Skip to content

Commit

Permalink
feat: support avs<>operator mapping with new APIs
Browse files Browse the repository at this point in the history
  • Loading branch information
bowenli86 committed Dec 4, 2023
1 parent 87f980e commit c4cd167
Show file tree
Hide file tree
Showing 5 changed files with 170 additions and 1 deletion.
86 changes: 86 additions & 0 deletions src/contracts/core/DelegationManager.sol
Original file line number Diff line number Diff line change
Expand Up @@ -450,6 +450,68 @@ contract DelegationManager is Initializable, OwnableUpgradeable, Pausable, Deleg
}
}

/**
* @notice Called by an avs to register an operator with the avs.
* @param operator The address of the operator to register.
* @param operatorSignature The signature, salt, and expiry of the operator's signature.
*/
function registerOperatorToAVS(
address operator,
ISignatureUtils.SignatureWithSaltAndExpiry memory operatorSignature) external {

require(
operatorSignature.expiry >= block.timestamp,
"DelegationManager.registerOperatorToAVS: operator signature expired"
);
require(
!operatorSaltIsSpent[operator][operatorSignature.salt],
"DelegationManager.registerOperatorToAVS: salt already spent"
);
require(
avsOperatorStatus[msg.sender][operator] == OperatorAVSRegistrationStatus.REGISTERED,
"DelegationManager.registerOperatorWithAVS: operator already registered"
);

// Calculate the digest hash
bytes32 operatorRegistrationDigestHash = calculateOperatorAVSRegistrationDigestHash({
operator: operator,
avs: msg.sender,
salt: operatorSignature.salt,
expiry: operatorSignature.expiry
});

// Check that the signature is valid
EIP1271SignatureUtils.checkSignature_EIP1271(
operator,
operatorRegistrationDigestHash,
operatorSignature.signature
);

// Mark the salt as spent
operatorSaltIsSpent[operator][operatorSignature.salt] = true;

// Set the operator as registered
avsOperatorStatus[msg.sender][operator] = OperatorAVSRegistrationStatus.REGISTERED;

emit OperatorAVSRegistrationStatusUpdated(operator, msg.sender, OperatorAVSRegistrationStatus.REGISTERED);
}

/**
* @notice Called by an avs to deregister an operator with the avs.
* @param operator The address of the operator to deregister.
*/
function deregisterOperatorFromAVS(address operator) external {
require(
avsOperatorStatus[msg.sender][operator] == OperatorAVSRegistrationStatus.REGISTERED,
"DelegationManager.deregisterOperatorFromAVS: operator not registered"
);

// Set the operator as deregistered
avsOperatorStatus[msg.sender][operator] = OperatorAVSRegistrationStatus.DEREGISTERED;

emit OperatorAVSRegistrationStatusUpdated(operator, msg.sender, OperatorAVSRegistrationStatus.DEREGISTERED);
}

/*******************************************************************************
INTERNAL FUNCTIONS
*******************************************************************************/
Expand Down Expand Up @@ -947,6 +1009,30 @@ contract DelegationManager is Initializable, OwnableUpgradeable, Pausable, Deleg
return approverDigestHash;
}

/**
* @notice Calculates the digest hash to be signed by an operator to register with an AVS
* @param operator The account registering as an operator
* @param avs The AVS the operator is registering to
* @param salt A unique and single use value associated with the approver signature.
* @param expiry Time after which the approver's signature becomes invalid
*/
function calculateOperatorAVSRegistrationDigestHash(
address operator,
address avs,
bytes32 salt,
uint256 expiry
) public view returns (bytes32) {
// calculate the struct hash
bytes32 structHash = keccak256(
abi.encode(OPERATOR_AVS_REGISTRATION_TYPEHASH, operator, avs, salt, expiry)
);
// calculate the digest hash
bytes32 digestHash = keccak256(
abi.encodePacked("\x19\x01", domainSeparator(), structHash)
);
return digestHash;
}

/**
* @dev Recalculates the domain separator when the chainid changes due to a fork.
*/
Expand Down
11 changes: 11 additions & 0 deletions src/contracts/core/DelegationManagerStorage.sol
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,10 @@ abstract contract DelegationManagerStorage is IDelegationManager {
bytes32 public constant DELEGATION_APPROVAL_TYPEHASH =
keccak256("DelegationApproval(address staker,address operator,bytes32 salt,uint256 expiry)");

/// @notice The EIP-712 typehash for the `Registration` struct used by the contract
bytes32 public constant OPERATOR_AVS_REGISTRATION_TYPEHASH =
keccak256("OperatorAVSRegistration(address operator,address avs,uint256 expiry)");

/**
* @notice Original EIP-712 Domain separator for this contract.
* @dev The domain separator may change in the event of a fork that modifies the ChainID.
Expand Down Expand Up @@ -92,6 +96,13 @@ abstract contract DelegationManagerStorage is IDelegationManager {
/// @notice the address of the StakeRegistry contract to call for stake updates when operator shares are changed
IStakeRegistryStub public stakeRegistry;

/// @notice Mapping: AVS => operator => enum of operator status to the AVS
mapping(address => mapping(address => OperatorAVSRegistrationStatus)) public avsOperatorStatus;

/// @notice Mapping: operator => 32-byte salt => whether or not the salt has already been used by the operator.
/// @dev Salt is used in the `registerOperatorToAVS` function.
mapping(address => mapping(bytes32 => bool)) public operatorSaltIsSpent;

constructor(IStrategyManager _strategyManager, ISlasher _slasher, IEigenPodManager _eigenPodManager) {
strategyManager = _strategyManager;
eigenPodManager = _eigenPodManager;
Expand Down
55 changes: 55 additions & 0 deletions src/contracts/interfaces/IDelegationManager.sol
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,12 @@ interface IDelegationManager is ISignatureUtils {
address withdrawer;
}

/// @notice Enum representing the status of an operator's registration with an AVS
enum OperatorAVSRegistrationStatus {
UNREGISTERED, // Operator not registered to AVS
REGISTERED // Operator registered to AVS
}

// @notice Emitted when a new operator registers in EigenLayer and provides their OperatorDetails.
event OperatorRegistered(address indexed operator, OperatorDetails operatorDetails);

Expand All @@ -122,6 +128,9 @@ interface IDelegationManager is ISignatureUtils {
*/
event AVSMetadataURIUpdated(address indexed avs, string metadataURI);

/// @notice Emitted when an operator's registration status for an AVS is updated
event OperatorAVSRegistrationStatusUpdated(address indexed operator, address indexed avs, OperatorAVSRegistrationStatus status);

/// @notice Emitted whenever an operator's shares are increased for a given strategy. Note that shares is the delta in the operator's shares.
event OperatorSharesIncreased(address indexed operator, address staker, IStrategy strategy, uint256 shares);

Expand Down Expand Up @@ -324,6 +333,49 @@ interface IDelegationManager is ISignatureUtils {
uint256 shares
) external;

/**
* @notice Called by an avs to register an operator with the avs.
* @param operator The address of the operator to register.
* @param operatorSignature The signature, salt, and expiry of the operator's signature.
*/
function registerOperatorToAVS(
address operator,
ISignatureUtils.SignatureWithSaltAndExpiry memory signatureWithSaltAndExpiry
) external;

/**
* @notice Called by an avs to deregister an operator with the avs.
* @param operator The address of the operator to deregister.
*/
function deregisterOperatorFromAVS(address operator) external;

/**
* @notice Returns whether or not an operator is registered to an avs.
* @param operator The address of the operator.
* @param avs The address of the avs.
*/
function registeredWithAVS(address operator, address avs) external view returns (bool);

/**
* @notice Returns whether or not the salt has already been used by the operator.
* @dev Salts is used in the `registerOperatorToAVS` function.
*/
function operatorSaltIsSpent(address avs, bytes32 salt) external view returns (bool);

/**
* @notice Calculates the digest hash to be signed by an operator to register with an AVS
* @param operator The account registering as an operator
* @param avs The AVS the operator is registering to
* @param salt A unique and single use value associated with the approver signature.
* @param expiry Time after which the approver's signature becomes invalid
*/
function calculateOperatorAVSRegistrationDigestHash(
address operator,
address avs,
bytes32 salt,
uint256 expiry
) external view returns (bytes32);

/// @notice the address of the StakeRegistry contract to call for stake updates when operator shares are changed
function stakeRegistry() external view returns (IStakeRegistryStub);

Expand Down Expand Up @@ -442,6 +494,9 @@ interface IDelegationManager is ISignatureUtils {
/// @notice The EIP-712 typehash for the DelegationApproval struct used by the contract
function DELEGATION_APPROVAL_TYPEHASH() external view returns (bytes32);

/// @notice The EIP-712 typehash for the Registration struct used by the contract
function OPERATOR_AVS_REGISTRATION_TYPEHASH() external view returns (bytes32);

/**
* @notice Getter function for the current EIP-712 domain separator for this contract.
*
Expand Down
3 changes: 3 additions & 0 deletions src/test/events/IDelegationManagerEvents.sol
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@ interface IDelegationManagerEvents {
*/
event AVSMetadataURIUpdated(address indexed avs, string metadataURI);

/// @notice Emitted when an operator's registration status for an AVS is updated
event OperatorAVSRegistrationStatusUpdated(address indexed operator, address indexed avs, OperatorAVSRegistrationStatus status);

/// @notice Emitted whenever an operator's shares are increased for a given strategy
event OperatorSharesIncreased(address indexed operator, address staker, IStrategy strategy, uint256 shares);

Expand Down
16 changes: 15 additions & 1 deletion src/test/mocks/DelegationManagerMock.sol
Original file line number Diff line number Diff line change
Expand Up @@ -109,20 +109,34 @@ contract DelegationManagerMock is IDelegationManager, Test {
function calculateStakerDigestHash(address /*staker*/, address /*operator*/, uint256 /*expiry*/)
external pure returns (bytes32 stakerDigestHash) {}

function calculateApproverDigestHash(address /*staker*/, address /*operator*/, uint256 /*expiry*/) external pure returns (bytes32 approverDigestHash) {}
function calculateApproverDigestHash(address /*staker*/, address /*operator*/, uint256 /*expiry*/)
external pure returns (bytes32 approverDigestHash) {}

function calculateOperatorAVSRegistrationDigestHash(address /*operator*/, address /*avs*/, bytes32 /*salt*/, uint256 /*expiry*/)
external pure returns (bytes32 digestHash) {}

function DOMAIN_TYPEHASH() external view returns (bytes32) {}

function STAKER_DELEGATION_TYPEHASH() external view returns (bytes32) {}

function DELEGATION_APPROVAL_TYPEHASH() external view returns (bytes32) {}

function OPERATOR_AVS_REGISTRATION_TYPEHASH() external view returns (bytes32) {}

function domainSeparator() external view returns (bytes32) {}

function cumulativeWithdrawalsQueued(address staker) external view returns (uint256) {}

function calculateWithdrawalRoot(Withdrawal memory withdrawal) external pure returns (bytes32) {}

function registerOperatorToAVS(address operator, ISignatureUtils.SignatureWithSaltAndExpiry memory operatorSignature) external {}

function deregisterOperatorFromAVS(address operator) external {}

function registeredWithAVS(address operator, address avs) external view returns (bool) {}

function operatorSaltIsSpent(address avs, bytes32 salt) external view returns (bool) {}

function queueWithdrawals(
QueuedWithdrawalParams[] calldata queuedWithdrawalParams
) external returns (bytes32[] memory) {}
Expand Down

0 comments on commit c4cd167

Please sign in to comment.