Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 1 internal transaction
Advanced mode:
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
14867767 | 921 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Minimal Proxy Contract for 0x85388a8cd772b19a468f982dc264c238856939c9
Contract Name:
Roles
Compiler Version
v0.8.6+commit.11564f7e
Optimization Enabled:
No with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity ^0.8.6; import "@gnosis.pm/zodiac/contracts/core/Modifier.sol"; import "./Permissions.sol"; contract Roles is Modifier { address public multisend; mapping(address => uint16) public defaultRoles; mapping(uint16 => Role) internal roles; event AssignRoles(address module, uint16[] roles, bool[] memberOf); event SetMultisendAddress(address multisendAddress); event RolesModSetup( address indexed initiator, address indexed owner, address indexed avatar, address target ); event SetDefaultRole(address module, uint16 defaultRole); /// `setUpModules` has already been called error SetUpModulesAlreadyCalled(); /// Arrays must be the same length error ArraysDifferentLength(); /// Sender is not a member of the role error NoMembership(); /// Sender is allowed to make this call, but the internal transaction failed error ModuleTransactionFailed(); /// @param _owner Address of the owner /// @param _avatar Address of the avatar (e.g. a Gnosis Safe) /// @param _target Address of the contract that will call exec function constructor( address _owner, address _avatar, address _target ) { bytes memory initParams = abi.encode(_owner, _avatar, _target); setUp(initParams); } function setUp(bytes memory initParams) public override { (address _owner, address _avatar, address _target) = abi.decode( initParams, (address, address, address) ); __Ownable_init(); avatar = _avatar; target = _target; transferOwnership(_owner); setupModules(); emit RolesModSetup(msg.sender, _owner, _avatar, _target); } function setupModules() internal { if (modules[SENTINEL_MODULES] != address(0)) { revert SetUpModulesAlreadyCalled(); } modules[SENTINEL_MODULES] = SENTINEL_MODULES; } /// @dev Set the address of the expected multisend library /// @notice Only callable by owner. /// @param _multisend address of the multisend library contract function setMultisend(address _multisend) external onlyOwner { multisend = _multisend; emit SetMultisendAddress(multisend); } /// @dev Allows all calls made to an address. /// @notice Only callable by owner. /// @param role Role to set for /// @param targetAddress Address to be allowed /// @param options defines whether or not delegate calls and/or eth can be sent to the target address. function allowTarget( uint16 role, address targetAddress, ExecutionOptions options ) external onlyOwner { Permissions.allowTarget(roles[role], role, targetAddress, options); } /// @dev Disallows all calls made to an address. /// @notice Only callable by owner. /// @param role Role to set for /// @param targetAddress Address to be disallowed function revokeTarget(uint16 role, address targetAddress) external onlyOwner { Permissions.revokeTarget(roles[role], role, targetAddress); } /// @dev Scopes calls to an address, limited to specific function signatures, and per function scoping rules. /// @notice Only callable by owner. /// @param role Role to set for. /// @param targetAddress Address to be scoped. function scopeTarget(uint16 role, address targetAddress) external onlyOwner { Permissions.scopeTarget(roles[role], role, targetAddress); } /// @dev Allows a specific function signature on a scoped target. /// @notice Only callable by owner. /// @param role Role to set for /// @param targetAddress Scoped address on which a function signature should be allowed. /// @param functionSig Function signature to be allowed. /// @param options Defines whether or not delegate calls and/or eth can be sent to the function. function scopeAllowFunction( uint16 role, address targetAddress, bytes4 functionSig, ExecutionOptions options ) external onlyOwner { Permissions.scopeAllowFunction( roles[role], role, targetAddress, functionSig, options ); } /// @dev Disallows a specific function signature on a scoped target. /// @notice Only callable by owner. /// @param role Role to set for /// @param targetAddress Scoped address on which a function signature should be disallowed. /// @param functionSig Function signature to be disallowed. function scopeRevokeFunction( uint16 role, address targetAddress, bytes4 functionSig ) external onlyOwner { Permissions.scopeRevokeFunction( roles[role], role, targetAddress, functionSig ); } /// @dev Sets scoping rules for a function, on a scoped address. /// @notice Only callable by owner. /// @param role Role to set for. /// @param targetAddress Scoped address on which scoping rules for a function are to be set. /// @param functionSig Function signature to be scoped. /// @param isParamScoped false for un-scoped, true for scoped. /// @param paramType Static, Dynamic or Dynamic32, depending on the parameter type. /// @param paramComp Any, or EqualTo, GreaterThan, or LessThan, depending on comparison type. /// @param compValue The reference value used while comparing and authorizing. /// @param options Defines whether or not delegate calls and/or eth can be sent to the function. function scopeFunction( uint16 role, address targetAddress, bytes4 functionSig, bool[] calldata isParamScoped, ParameterType[] calldata paramType, Comparison[] calldata paramComp, bytes[] memory compValue, ExecutionOptions options ) external onlyOwner { Permissions.scopeFunction( roles[role], role, targetAddress, functionSig, isParamScoped, paramType, paramComp, compValue, options ); } /// @dev Sets whether or not delegate calls and/or eth can be sent to a function on a scoped target. /// @notice Only callable by owner. /// @notice Only in play when target is scoped. /// @param role Role to set for. /// @param targetAddress Scoped address on which the ExecutionOptions for a function are to be set. /// @param functionSig Function signature on which the ExecutionOptions are to be set. /// @param options Defines whether or not delegate calls and/or eth can be sent to the function. function scopeFunctionExecutionOptions( uint16 role, address targetAddress, bytes4 functionSig, ExecutionOptions options ) external onlyOwner { Permissions.scopeFunctionExecutionOptions( roles[role], role, targetAddress, functionSig, options ); } /// @dev Sets and enforces scoping rules, for a single parameter of a function, on a scoped target. /// @notice Only callable by owner. /// @param role Role to set for. /// @param targetAddress Scoped address on which functionSig lives. /// @param functionSig Function signature to be scoped. /// @param paramIndex The index of the parameter to scope. /// @param paramType Static, Dynamic or Dynamic32, depending on the parameter type. /// @param paramComp Any, or EqualTo, GreaterThan, or LessThan, depending on comparison type. /// @param compValue The reference value used while comparing and authorizing. function scopeParameter( uint16 role, address targetAddress, bytes4 functionSig, uint256 paramIndex, ParameterType paramType, Comparison paramComp, bytes calldata compValue ) external onlyOwner { Permissions.scopeParameter( roles[role], role, targetAddress, functionSig, paramIndex, paramType, paramComp, compValue ); } /// @dev Sets and enforces scoping rules, for a single parameter of a function, on a scoped target. /// @notice Only callable by owner. /// @notice Parameter will be scoped with comparison type OneOf. /// @param role Role to set for. /// @param targetAddress Scoped address on which functionSig lives. /// @param functionSig Function signature to be scoped. /// @param paramIndex The index of the parameter to scope. /// @param paramType Static, Dynamic or Dynamic32, depending on the parameter type. /// @param compValues The reference values used while comparing and authorizing. function scopeParameterAsOneOf( uint16 role, address targetAddress, bytes4 functionSig, uint256 paramIndex, ParameterType paramType, bytes[] calldata compValues ) external onlyOwner { Permissions.scopeParameterAsOneOf( roles[role], role, targetAddress, functionSig, paramIndex, paramType, compValues ); } /// @dev Un-scopes a single parameter of a function, on a scoped target. /// @notice Only callable by owner. /// @param role Role to set for. /// @param targetAddress Scoped address on which functionSig lives. /// @param functionSig Function signature to be scoped. /// @param paramIndex The index of the parameter to un-scope. function unscopeParameter( uint16 role, address targetAddress, bytes4 functionSig, uint8 paramIndex ) external onlyOwner { Permissions.unscopeParameter( roles[role], role, targetAddress, functionSig, paramIndex ); } /// @dev Assigns and revokes roles to a given module. /// @param module Module on which to assign/revoke roles. /// @param _roles Roles to assign/revoke. /// @param memberOf Assign (true) or revoke (false) corresponding _roles. function assignRoles( address module, uint16[] calldata _roles, bool[] calldata memberOf ) external onlyOwner { if (_roles.length != memberOf.length) { revert ArraysDifferentLength(); } for (uint16 i = 0; i < _roles.length; i++) { roles[_roles[i]].members[module] = memberOf[i]; } if (!isModuleEnabled(module)) { enableModule(module); } emit AssignRoles(module, _roles, memberOf); } /// @dev Sets the default role used for a module if it calls execTransactionFromModule() or execTransactionFromModuleReturnData(). /// @param module Address of the module on which to set default role. /// @param role Role to be set as default. function setDefaultRole(address module, uint16 role) external onlyOwner { defaultRoles[module] = role; emit SetDefaultRole(module, role); } /// @dev Passes a transaction to the modifier. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @notice Can only be called by enabled modules function execTransactionFromModule( address to, uint256 value, bytes calldata data, Enum.Operation operation ) public override moduleOnly returns (bool success) { Permissions.check( roles[defaultRoles[msg.sender]], multisend, to, value, data, operation ); return exec(to, value, data, operation); } /// @dev Passes a transaction to the modifier, expects return data. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @notice Can only be called by enabled modules function execTransactionFromModuleReturnData( address to, uint256 value, bytes calldata data, Enum.Operation operation ) public override moduleOnly returns (bool, bytes memory) { Permissions.check( roles[defaultRoles[msg.sender]], multisend, to, value, data, operation ); return execAndReturnData(to, value, data, operation); } /// @dev Passes a transaction to the modifier assuming the specified role. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @param role Identifier of the role to assume for this transaction /// @param shouldRevert Should the function revert on inner execution returning success false? /// @notice Can only be called by enabled modules function execTransactionWithRole( address to, uint256 value, bytes calldata data, Enum.Operation operation, uint16 role, bool shouldRevert ) public moduleOnly returns (bool success) { Permissions.check(roles[role], multisend, to, value, data, operation); success = exec(to, value, data, operation); if (shouldRevert && !success) { revert ModuleTransactionFailed(); } } /// @dev Passes a transaction to the modifier assuming the specified role. Expects return data. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @param role Identifier of the role to assume for this transaction /// @param shouldRevert Should the function revert on inner execution returning success false? /// @notice Can only be called by enabled modules function execTransactionWithRoleReturnData( address to, uint256 value, bytes calldata data, Enum.Operation operation, uint16 role, bool shouldRevert ) public moduleOnly returns (bool success, bytes memory returnData) { Permissions.check(roles[role], multisend, to, value, data, operation); (success, returnData) = execAndReturnData(to, value, data, operation); if (shouldRevert && !success) { revert ModuleTransactionFailed(); } } }
// SPDX-License-Identifier: LGPL-3.0-only /// @title Modifier Interface - A contract that sits between a Aodule and an Avatar and enforce some additional logic. pragma solidity >=0.7.0 <0.9.0; import "../interfaces/IAvatar.sol"; import "./Module.sol"; abstract contract Modifier is Module { event EnabledModule(address module); event DisabledModule(address module); address internal constant SENTINEL_MODULES = address(0x1); // Mapping of modules mapping(address => address) internal modules; /* -------------------------------------------------- You must override at least one of following two virtual functions, execTransactionFromModule() and execTransactionFromModuleReturnData(). */ /// @dev Passes a transaction to the modifier. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @notice Can only be called by enabled modules function execTransactionFromModule( address to, uint256 value, bytes calldata data, Enum.Operation operation ) public virtual moduleOnly returns (bool success) {} /// @dev Passes a transaction to the modifier, expects return data. /// @param to Destination address of module transaction /// @param value Ether value of module transaction /// @param data Data payload of module transaction /// @param operation Operation type of module transaction /// @notice Can only be called by enabled modules function execTransactionFromModuleReturnData( address to, uint256 value, bytes calldata data, Enum.Operation operation ) public virtual moduleOnly returns (bool success, bytes memory returnData) {} /* -------------------------------------------------- */ modifier moduleOnly() { require(modules[msg.sender] != address(0), "Module not authorized"); _; } /// @dev Disables a module on the modifier /// @param prevModule Module that pointed to the module to be removed in the linked list /// @param module Module to be removed /// @notice This can only be called by the owner function disableModule(address prevModule, address module) public onlyOwner { require( module != address(0) && module != SENTINEL_MODULES, "Invalid module" ); require(modules[prevModule] == module, "Module already disabled"); modules[prevModule] = modules[module]; modules[module] = address(0); emit DisabledModule(module); } /// @dev Enables a module that can add transactions to the queue /// @param module Address of the module to be enabled /// @notice This can only be called by the owner function enableModule(address module) public onlyOwner { require( module != address(0) && module != SENTINEL_MODULES, "Invalid module" ); require(modules[module] == address(0), "Module already enabled"); modules[module] = modules[SENTINEL_MODULES]; modules[SENTINEL_MODULES] = module; emit EnabledModule(module); } /// @dev Returns if an module is enabled /// @return True if the module is enabled function isModuleEnabled(address _module) public view returns (bool) { return SENTINEL_MODULES != _module && modules[_module] != address(0); } /// @dev Returns array of modules. /// @param start Start of the page. /// @param pageSize Maximum number of modules that should be returned. /// @return array Array of modules. /// @return next Start of the next page. function getModulesPaginated(address start, uint256 pageSize) external view returns (address[] memory array, address next) { // Init array with max page size array = new address[](pageSize); // Populate return array uint256 moduleCount = 0; address currentModule = modules[start]; while ( currentModule != address(0x0) && currentModule != SENTINEL_MODULES && moduleCount < pageSize ) { array[moduleCount] = currentModule; currentModule = modules[currentModule]; moduleCount++; } next = currentModule; // Set correct size of returned array // solhint-disable-next-line no-inline-assembly assembly { mstore(array, moduleCount) } } }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity ^0.8.6; import "@gnosis.pm/safe-contracts/contracts/common/Enum.sol"; enum ParameterType { Static, Dynamic, Dynamic32 } enum Comparison { EqualTo, GreaterThan, LessThan, OneOf } enum ExecutionOptions { None, Send, DelegateCall, Both } enum Clearance { None, Target, Function } struct TargetAddress { Clearance clearance; ExecutionOptions options; } struct Role { mapping(address => bool) members; mapping(address => TargetAddress) targets; mapping(bytes32 => uint256) functions; mapping(bytes32 => bytes32) compValues; mapping(bytes32 => bytes32[]) compValuesOneOf; } library Permissions { uint256 internal constant SCOPE_MAX_PARAMS = 48; event AllowTarget( uint16 role, address targetAddress, ExecutionOptions options ); event RevokeTarget(uint16 role, address targetAddress); event ScopeTarget(uint16 role, address targetAddress); event ScopeAllowFunction( uint16 role, address targetAddress, bytes4 selector, ExecutionOptions options, uint256 resultingScopeConfig ); event ScopeRevokeFunction( uint16 role, address targetAddress, bytes4 selector, uint256 resultingScopeConfig ); event ScopeFunction( uint16 role, address targetAddress, bytes4 functionSig, bool[] isParamScoped, ParameterType[] paramType, Comparison[] paramComp, bytes[] compValue, ExecutionOptions options, uint256 resultingScopeConfig ); event ScopeFunctionExecutionOptions( uint16 role, address targetAddress, bytes4 functionSig, ExecutionOptions options, uint256 resultingScopeConfig ); event ScopeParameter( uint16 role, address targetAddress, bytes4 functionSig, uint256 index, ParameterType paramType, Comparison paramComp, bytes compValue, uint256 resultingScopeConfig ); event ScopeParameterAsOneOf( uint16 role, address targetAddress, bytes4 functionSig, uint256 index, ParameterType paramType, bytes[] compValues, uint256 resultingScopeConfig ); event UnscopeParameter( uint16 role, address targetAddress, bytes4 functionSig, uint256 index, uint256 resultingScopeConfig ); /// Sender is not a member of the role error NoMembership(); /// Arrays must be the same length error ArraysDifferentLength(); /// Function signature too short error FunctionSignatureTooShort(); /// Role not allowed to delegate call to target address error DelegateCallNotAllowed(); /// Role not allowed to call target address error TargetAddressNotAllowed(); /// Role not allowed to call this function on target address error FunctionNotAllowed(); /// Role not allowed to send to target address error SendNotAllowed(); /// Role not allowed to use bytes for parameter error ParameterNotAllowed(); /// Role not allowed to use bytes for parameter error ParameterNotOneOfAllowed(); /// Role not allowed to use bytes less than value for parameter error ParameterLessThanAllowed(); /// Role not allowed to use bytes greater than value for parameter error ParameterGreaterThanAllowed(); /// only multisend txs with an offset of 32 bytes are allowed error UnacceptableMultiSendOffset(); /// OneOf Comparison must be set via dedicated function error UnsuitableOneOfComparison(); /// Not possible to define gt/lt for Dynamic types error UnsuitableRelativeComparison(); /// CompValue for static types should have a size of exactly 32 bytes error UnsuitableStaticCompValueSize(); /// CompValue for Dynamic32 types should be a multiple of exactly 32 bytes error UnsuitableDynamic32CompValueSize(); /// Exceeds the max number of params supported error ScopeMaxParametersExceeded(); /// OneOf Comparison requires at least two compValues error NotEnoughCompValuesForOneOf(); /// The provided calldata for execution is too short, or an OutOfBounds scoped parameter was configured error CalldataOutOfBounds(); /* * * CHECKERS * */ function check( Role storage role, address multisend, address to, uint256 value, bytes calldata data, Enum.Operation operation ) public view { if (!role.members[msg.sender]) { revert NoMembership(); } if (multisend == to) { checkMultisendTransaction(role, data); } else { checkTransaction(role, to, value, data, operation); } } /// @dev Splits a multisend data blob into transactions and forwards them to be checked. /// @param data the packed transaction data (created by utils function buildMultiSendSafeTx). /// @param role Role to check for. function checkMultisendTransaction(Role storage role, bytes memory data) internal view { Enum.Operation operation; address to; uint256 value; bytes memory out; uint256 dataLength; uint256 offset; assembly { offset := mload(add(data, 36)) } if (offset != 32) { revert UnacceptableMultiSendOffset(); } // transaction data (1st tx operation) reads at byte 100, // 4 bytes (multisend_id) + 32 bytes (offset_multisend_data) + 32 bytes multisend_data_length // increment i by the transaction data length // + 85 bytes of the to, value, and operation bytes until we reach the end of the data for (uint256 i = 100; i < data.length; i += (85 + dataLength)) { assembly { // First byte of the data is the operation. // We shift by 248 bits (256 - 8 [operation byte]) right since mload will always load 32 bytes (a word). // This will also zero out unused data. operation := shr(0xf8, mload(add(data, i))) // We offset the load address by 1 byte (operation byte) // We shift it right by 96 bits (256 - 160 [20 address bytes]) to right-align the data and zero out unused data. to := shr(0x60, mload(add(data, add(i, 0x01)))) // We offset the load address by 21 byte (operation byte + 20 address bytes) value := mload(add(data, add(i, 0x15))) // We offset the load address by 53 byte (operation byte + 20 address bytes + 32 value bytes) dataLength := mload(add(data, add(i, 0x35))) // We offset the load address by 85 byte (operation byte + 20 address bytes + 32 value bytes + 32 data length bytes) out := add(data, add(i, 0x35)) } checkTransaction(role, to, value, out, operation); } } function checkTransaction( Role storage role, address targetAddress, uint256 value, bytes memory data, Enum.Operation operation ) internal view { if (data.length != 0 && data.length < 4) { revert FunctionSignatureTooShort(); } TargetAddress storage target = role.targets[targetAddress]; if (target.clearance == Clearance.None) { revert TargetAddressNotAllowed(); } if (target.clearance == Clearance.Target) { checkExecutionOptions(value, operation, target.options); return; } if (target.clearance == Clearance.Function) { uint256 scopeConfig = role.functions[ keyForFunctions(targetAddress, bytes4(data)) ]; if (scopeConfig == 0) { revert FunctionNotAllowed(); } (ExecutionOptions options, bool isWildcarded, ) = unpackFunction( scopeConfig ); checkExecutionOptions(value, operation, options); if (isWildcarded == false) { checkParameters(role, scopeConfig, targetAddress, data); } return; } assert(false); } function checkExecutionOptions( uint256 value, Enum.Operation operation, ExecutionOptions options ) internal pure { // isSend && !canSend if ( value > 0 && options != ExecutionOptions.Send && options != ExecutionOptions.Both ) { revert SendNotAllowed(); } // isDelegateCall && !canDelegateCall if ( operation == Enum.Operation.DelegateCall && options != ExecutionOptions.DelegateCall && options != ExecutionOptions.Both ) { revert DelegateCallNotAllowed(); } } /// @dev Will revert if a transaction has a parameter that is not allowed /// @param role reference to role storage /// @param targetAddress Address to check. /// @param data the transaction data to check function checkParameters( Role storage role, uint256 scopeConfig, address targetAddress, bytes memory data ) internal view { bytes4 functionSig = bytes4(data); (, , uint256 length) = unpackFunction(scopeConfig); for (uint256 i = 0; i < length; i++) { ( bool isScoped, ParameterType paramType, Comparison paramComp ) = unpackParameter(scopeConfig, i); if (!isScoped) { continue; } bytes32 value; if (paramType != ParameterType.Static) { value = pluckDynamicValue(data, paramType, i); } else { value = pluckStaticValue(data, i); } bytes32 key = keyForCompValues(targetAddress, functionSig, i); if (paramComp != Comparison.OneOf) { compare(paramComp, role.compValues[key], value); } else { compareOneOf(role.compValuesOneOf[key], value); } } } function compare( Comparison paramComp, bytes32 compValue, bytes32 value ) internal pure { if (paramComp == Comparison.EqualTo && value != compValue) { revert ParameterNotAllowed(); } else if (paramComp == Comparison.GreaterThan && value <= compValue) { revert ParameterLessThanAllowed(); } else if (paramComp == Comparison.LessThan && value >= compValue) { revert ParameterGreaterThanAllowed(); } } function compareOneOf(bytes32[] storage compValue, bytes32 value) internal view { for (uint256 i = 0; i < compValue.length; i++) { if (value == compValue[i]) return; } revert ParameterNotOneOfAllowed(); } /* * * SETTERS * */ function allowTarget( Role storage role, uint16 roleId, address targetAddress, ExecutionOptions options ) external { role.targets[targetAddress] = TargetAddress(Clearance.Target, options); emit AllowTarget(roleId, targetAddress, options); } function revokeTarget( Role storage role, uint16 roleId, address targetAddress ) external { role.targets[targetAddress] = TargetAddress( Clearance.None, ExecutionOptions.None ); emit RevokeTarget(roleId, targetAddress); } function scopeTarget( Role storage role, uint16 roleId, address targetAddress ) external { role.targets[targetAddress] = TargetAddress( Clearance.Function, ExecutionOptions.None ); emit ScopeTarget(roleId, targetAddress); } function scopeAllowFunction( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, ExecutionOptions options ) external { /* * packLeft( * 0 -> start from a fresh scopeConfig * options -> externally provided options * true -> mark the function as wildcarded * 0 -> length * ) */ uint256 scopeConfig = packLeft(0, options, true, 0); role.functions[ keyForFunctions(targetAddress, functionSig) ] = scopeConfig; emit ScopeAllowFunction( roleId, targetAddress, functionSig, options, scopeConfig ); } function scopeRevokeFunction( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig ) external { role.functions[keyForFunctions(targetAddress, functionSig)] = 0; emit ScopeRevokeFunction(roleId, targetAddress, functionSig, 0); } function scopeFunction( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, bool[] memory isScoped, ParameterType[] memory paramType, Comparison[] memory paramComp, bytes[] calldata compValue, ExecutionOptions options ) external { uint256 length = isScoped.length; if ( length != paramType.length || length != paramComp.length || length != compValue.length ) { revert ArraysDifferentLength(); } if (length > SCOPE_MAX_PARAMS) { revert ScopeMaxParametersExceeded(); } for (uint256 i = 0; i < length; i++) { if (isScoped[i]) { enforceComp(paramType[i], paramComp[i]); enforceCompValue(paramType[i], compValue[i]); } } /* * packLeft( * 0 -> start from a fresh scopeConfig * options -> externally provided options * false -> mark the function as not wildcarded * 0 -> length * ) */ uint256 scopeConfig = packLeft(0, options, false, length); for (uint256 i = 0; i < length; i++) { scopeConfig = packRight( scopeConfig, i, isScoped[i], paramType[i], paramComp[i] ); } //set scopeConfig role.functions[ keyForFunctions(targetAddress, functionSig) ] = scopeConfig; //set compValues for (uint256 i = 0; i < length; i++) { role.compValues[ keyForCompValues(targetAddress, functionSig, i) ] = compressCompValue(paramType[i], compValue[i]); } emit ScopeFunction( roleId, targetAddress, functionSig, isScoped, paramType, paramComp, compValue, options, scopeConfig ); } function scopeFunctionExecutionOptions( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, ExecutionOptions options ) external { bytes32 key = keyForFunctions(targetAddress, functionSig); //set scopeConfig uint256 scopeConfig = packOptions(role.functions[key], options); role.functions[ keyForFunctions(targetAddress, functionSig) ] = scopeConfig; emit ScopeFunctionExecutionOptions( roleId, targetAddress, functionSig, options, scopeConfig ); } function scopeParameter( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, uint256 index, ParameterType paramType, Comparison paramComp, bytes calldata compValue ) external { if (index >= SCOPE_MAX_PARAMS) { revert ScopeMaxParametersExceeded(); } enforceComp(paramType, paramComp); enforceCompValue(paramType, compValue); // set scopeConfig bytes32 key = keyForFunctions(targetAddress, functionSig); uint256 scopeConfig = packParameter( role.functions[key], index, true, // isScoped paramType, paramComp ); role.functions[key] = scopeConfig; // set compValue role.compValues[ keyForCompValues(targetAddress, functionSig, index) ] = compressCompValue(paramType, compValue); emit ScopeParameter( roleId, targetAddress, functionSig, index, paramType, paramComp, compValue, scopeConfig ); } function scopeParameterAsOneOf( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, uint256 index, ParameterType paramType, bytes[] calldata compValues ) external { if (index >= SCOPE_MAX_PARAMS) { revert ScopeMaxParametersExceeded(); } if (compValues.length < 2) { revert NotEnoughCompValuesForOneOf(); } for (uint256 i = 0; i < compValues.length; i++) { enforceCompValue(paramType, compValues[i]); } // set scopeConfig bytes32 key = keyForFunctions(targetAddress, functionSig); uint256 scopeConfig = packParameter( role.functions[key], index, true, // isScoped paramType, Comparison.OneOf ); role.functions[key] = scopeConfig; // set compValue key = keyForCompValues(targetAddress, functionSig, index); role.compValuesOneOf[key] = new bytes32[](compValues.length); for (uint256 i = 0; i < compValues.length; i++) { role.compValuesOneOf[key][i] = compressCompValue( paramType, compValues[i] ); } emit ScopeParameterAsOneOf( roleId, targetAddress, functionSig, index, paramType, compValues, scopeConfig ); } function unscopeParameter( Role storage role, uint16 roleId, address targetAddress, bytes4 functionSig, uint256 index ) external { if (index >= SCOPE_MAX_PARAMS) { revert ScopeMaxParametersExceeded(); } // set scopeConfig bytes32 key = keyForFunctions(targetAddress, functionSig); uint256 scopeConfig = packParameter( role.functions[key], index, false, // isScoped ParameterType(0), Comparison(0) ); role.functions[key] = scopeConfig; emit UnscopeParameter( roleId, targetAddress, functionSig, index, scopeConfig ); } function enforceComp(ParameterType paramType, Comparison paramComp) internal pure { if (paramComp == Comparison.OneOf) { revert UnsuitableOneOfComparison(); } if ( (paramType != ParameterType.Static) && (paramComp != Comparison.EqualTo) ) { revert UnsuitableRelativeComparison(); } } function enforceCompValue(ParameterType paramType, bytes calldata compValue) internal pure { if (paramType == ParameterType.Static && compValue.length != 32) { revert UnsuitableStaticCompValueSize(); } if ( paramType == ParameterType.Dynamic32 && compValue.length % 32 != 0 ) { revert UnsuitableDynamic32CompValueSize(); } } /* * * HELPERS * */ function pluckDynamicValue( bytes memory data, ParameterType paramType, uint256 index ) internal pure returns (bytes32) { assert(paramType != ParameterType.Static); // pre-check: is there a word available for the current parameter at argumentsBlock? if (data.length < 4 + index * 32 + 32) { revert CalldataOutOfBounds(); } /* * Encoded calldata: * 4 bytes -> function selector * 32 bytes -> sequence, one chunk per parameter * * There is one (byte32) chunk per parameter. Depending on type it contains: * Static -> value encoded inline (not plucked by this function) * Dynamic -> a byte offset to encoded data payload * Dynamic32 -> a byte offset to encoded data payload * Note: Fixed Sized Arrays (e.g., bool[2]), are encoded inline * Note: Nested types also do not follow the above described rules, and are unsupported * Note: The offset to payload does not include 4 bytes for functionSig * * * At encoded payload, the first 32 bytes are the length encoding of the parameter payload. Depending on ParameterType: * Dynamic -> length in bytes * Dynamic32 -> length in bytes32 * Note: Dynamic types are: bytes, string * Note: Dynamic32 types are non-nested arrays: address[] bytes32[] uint[] etc */ // the start of the parameter block // 32 bytes - length encoding of the data bytes array // 4 bytes - function sig uint256 argumentsBlock; assembly { argumentsBlock := add(data, 36) } // the two offsets are relative to argumentsBlock uint256 offset = index * 32; uint256 offsetPayload; assembly { offsetPayload := mload(add(argumentsBlock, offset)) } uint256 lengthPayload; assembly { lengthPayload := mload(add(argumentsBlock, offsetPayload)) } // account for: // 4 bytes - functionSig // 32 bytes - length encoding for the parameter payload uint256 start = 4 + offsetPayload + 32; uint256 end = start + ( paramType == ParameterType.Dynamic32 ? lengthPayload * 32 : lengthPayload ); // are we slicing out of bounds? if (data.length < end) { revert CalldataOutOfBounds(); } return keccak256(slice(data, start, end)); } function pluckStaticValue(bytes memory data, uint256 index) internal pure returns (bytes32) { // pre-check: is there a word available for the current parameter at argumentsBlock? if (data.length < 4 + index * 32 + 32) { revert CalldataOutOfBounds(); } uint256 offset = 4 + index * 32; bytes32 value; assembly { // add 32 - jump over the length encoding of the data bytes array value := mload(add(32, add(data, offset))) } return value; } function slice( bytes memory data, uint256 start, uint256 end ) internal pure returns (bytes memory result) { result = new bytes(end - start); for (uint256 j = start; j < end; j++) { result[j - start] = data[j]; } } /* * pack/unpack are bit helpers for scopeConfig */ function packParameter( uint256 scopeConfig, uint256 index, bool isScoped, ParameterType paramType, Comparison paramComp ) internal pure returns (uint256) { (ExecutionOptions options, , uint256 prevLength) = unpackFunction( scopeConfig ); uint256 nextLength = index + 1 > prevLength ? index + 1 : prevLength; return packLeft( packRight(scopeConfig, index, isScoped, paramType, paramComp), options, false, // isWildcarded=false nextLength ); } function packOptions(uint256 scopeConfig, ExecutionOptions options) internal pure returns (uint256) { uint256 optionsMask = 3 << 254; scopeConfig &= ~optionsMask; scopeConfig |= uint256(options) << 254; return scopeConfig; } function packLeft( uint256 scopeConfig, ExecutionOptions options, bool isWildcarded, uint256 length ) internal pure returns (uint256) { // LEFT SIDE // 2 bits -> options // 1 bits -> isWildcarded // 5 bits -> unused // 8 bits -> length // RIGHT SIDE // 48 bits -> isScoped // 96 bits -> paramType (2 bits per entry 48*2) // 96 bits -> paramComp (2 bits per entry 48*2) // Wipe the LEFT SIDE clean. Start from there scopeConfig = (scopeConfig << 16) >> 16; // set options -> 256 - 2 = 254 scopeConfig |= uint256(options) << 254; // set isWildcarded -> 256 - 2 - 1 = 253 if (isWildcarded) { scopeConfig |= 1 << 253; } // set Length -> 48 + 96 + 96 = 240 scopeConfig |= length << 240; return scopeConfig; } function packRight( uint256 scopeConfig, uint256 index, bool isScoped, ParameterType paramType, Comparison paramComp ) internal pure returns (uint256) { // LEFT SIDE // 2 bits -> options // 1 bits -> isWildcarded // 5 bits -> unused // 8 bits -> length // RIGHT SIDE // 48 bits -> isScoped // 96 bits -> paramType (2 bits per entry 48*2) // 96 bits -> paramComp (2 bits per entry 48*2) uint256 isScopedMask = 1 << (index + 96 + 96); uint256 paramTypeMask = 3 << (index * 2 + 96); uint256 paramCompMask = 3 << (index * 2); if (isScoped) { scopeConfig |= isScopedMask; } else { scopeConfig &= ~isScopedMask; } scopeConfig &= ~paramTypeMask; scopeConfig |= uint256(paramType) << (index * 2 + 96); scopeConfig &= ~paramCompMask; scopeConfig |= uint256(paramComp) << (index * 2); return scopeConfig; } function unpackFunction(uint256 scopeConfig) internal pure returns ( ExecutionOptions options, bool isWildcarded, uint256 length ) { uint256 isWildcardedMask = 1 << 253; options = ExecutionOptions(scopeConfig >> 254); isWildcarded = scopeConfig & isWildcardedMask != 0; length = (scopeConfig << 8) >> 248; } function unpackParameter(uint256 scopeConfig, uint256 index) internal pure returns ( bool isScoped, ParameterType paramType, Comparison paramComp ) { uint256 isScopedMask = 1 << (index + 96 + 96); uint256 paramTypeMask = 3 << (index * 2 + 96); uint256 paramCompMask = 3 << (index * 2); isScoped = (scopeConfig & isScopedMask) != 0; paramType = ParameterType( (scopeConfig & paramTypeMask) >> (index * 2 + 96) ); paramComp = Comparison((scopeConfig & paramCompMask) >> (index * 2)); } function keyForFunctions(address targetAddress, bytes4 functionSig) public pure returns (bytes32) { return bytes32(abi.encodePacked(targetAddress, functionSig)); } function keyForCompValues( address targetAddress, bytes4 functionSig, uint256 index ) public pure returns (bytes32) { return bytes32(abi.encodePacked(targetAddress, functionSig, uint8(index))); } function compressCompValue( ParameterType paramType, bytes calldata compValue ) internal pure returns (bytes32) { return paramType == ParameterType.Static ? bytes32(compValue) : keccak256(compValue); } }
// SPDX-License-Identifier: LGPL-3.0-only /// @title Zodiac Avatar - A contract that manages modules that can execute transactions via this contract. pragma solidity >=0.7.0 <0.9.0; import "@gnosis.pm/safe-contracts/contracts/common/Enum.sol"; interface IAvatar { /// @dev Enables a module on the avatar. /// @notice Can only be called by the avatar. /// @notice Modules should be stored as a linked list. /// @notice Must emit EnabledModule(address module) if successful. /// @param module Module to be enabled. function enableModule(address module) external; /// @dev Disables a module on the avatar. /// @notice Can only be called by the avatar. /// @notice Must emit DisabledModule(address module) if successful. /// @param prevModule Address that pointed to the module to be removed in the linked list /// @param module Module to be removed. function disableModule(address prevModule, address module) external; /// @dev Allows a Module to execute a transaction. /// @notice Can only be called by an enabled module. /// @notice Must emit ExecutionFromModuleSuccess(address module) if successful. /// @notice Must emit ExecutionFromModuleFailure(address module) if unsuccessful. /// @param to Destination address of module transaction. /// @param value Ether value of module transaction. /// @param data Data payload of module transaction. /// @param operation Operation type of module transaction: 0 == call, 1 == delegate call. function execTransactionFromModule( address to, uint256 value, bytes memory data, Enum.Operation operation ) external returns (bool success); /// @dev Allows a Module to execute a transaction and return data /// @notice Can only be called by an enabled module. /// @notice Must emit ExecutionFromModuleSuccess(address module) if successful. /// @notice Must emit ExecutionFromModuleFailure(address module) if unsuccessful. /// @param to Destination address of module transaction. /// @param value Ether value of module transaction. /// @param data Data payload of module transaction. /// @param operation Operation type of module transaction: 0 == call, 1 == delegate call. function execTransactionFromModuleReturnData( address to, uint256 value, bytes memory data, Enum.Operation operation ) external returns (bool success, bytes memory returnData); /// @dev Returns if an module is enabled /// @return True if the module is enabled function isModuleEnabled(address module) external view returns (bool); /// @dev Returns array of modules. /// @param start Start of the page. /// @param pageSize Maximum number of modules that should be returned. /// @return array Array of modules. /// @return next Start of the next page. function getModulesPaginated(address start, uint256 pageSize) external view returns (address[] memory array, address next); }
// SPDX-License-Identifier: LGPL-3.0-only /// @title Module Interface - A contract that can pass messages to a Module Manager contract if enabled by that contract. pragma solidity >=0.7.0 <0.9.0; import "../interfaces/IAvatar.sol"; import "../factory/FactoryFriendly.sol"; import "../guard/Guardable.sol"; abstract contract Module is FactoryFriendly, Guardable { /// @dev Emitted each time the avatar is set. event AvatarSet(address indexed previousAvatar, address indexed newAvatar); /// @dev Emitted each time the Target is set. event TargetSet(address indexed previousTarget, address indexed newTarget); /// @dev Address that will ultimately execute function calls. address public avatar; /// @dev Address that this module will pass transactions to. address public target; /// @dev Sets the avatar to a new avatar (`newAvatar`). /// @notice Can only be called by the current owner. function setAvatar(address _avatar) public onlyOwner { address previousAvatar = avatar; avatar = _avatar; emit AvatarSet(previousAvatar, _avatar); } /// @dev Sets the target to a new target (`newTarget`). /// @notice Can only be called by the current owner. function setTarget(address _target) public onlyOwner { address previousTarget = target; target = _target; emit TargetSet(previousTarget, _target); } /// @dev Passes a transaction to be executed by the avatar. /// @notice Can only be called by this contract. /// @param to Destination address of module transaction. /// @param value Ether value of module transaction. /// @param data Data payload of module transaction. /// @param operation Operation type of module transaction: 0 == call, 1 == delegate call. function exec( address to, uint256 value, bytes memory data, Enum.Operation operation ) internal returns (bool success) { /// check if a transactioon guard is enabled. if (guard != address(0)) { IGuard(guard).checkTransaction( /// Transaction info used by module transactions to, value, data, operation, /// Zero out the redundant transaction information only used for Safe multisig transctions 0, 0, 0, address(0), payable(0), bytes("0x"), address(0) ); } success = IAvatar(target).execTransactionFromModule( to, value, data, operation ); if (guard != address(0)) { IGuard(guard).checkAfterExecution(bytes32("0x"), success); } return success; } /// @dev Passes a transaction to be executed by the target and returns data. /// @notice Can only be called by this contract. /// @param to Destination address of module transaction. /// @param value Ether value of module transaction. /// @param data Data payload of module transaction. /// @param operation Operation type of module transaction: 0 == call, 1 == delegate call. function execAndReturnData( address to, uint256 value, bytes memory data, Enum.Operation operation ) internal returns (bool success, bytes memory returnData) { /// check if a transactioon guard is enabled. if (guard != address(0)) { IGuard(guard).checkTransaction( /// Transaction info used by module transactions to, value, data, operation, /// Zero out the redundant transaction information only used for Safe multisig transctions 0, 0, 0, address(0), payable(0), bytes("0x"), address(0) ); } (success, returnData) = IAvatar(target) .execTransactionFromModuleReturnData(to, value, data, operation); if (guard != address(0)) { IGuard(guard).checkAfterExecution(bytes32("0x"), success); } return (success, returnData); } }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity >=0.7.0 <0.9.0; /// @title Enum - Collection of enums /// @author Richard Meissner - <[email protected]> contract Enum { enum Operation {Call, DelegateCall} }
// SPDX-License-Identifier: LGPL-3.0-only /// @title Zodiac FactoryFriendly - A contract that allows other contracts to be initializable and pass bytes as arguments to define contract state pragma solidity >=0.7.0 <0.9.0; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; abstract contract FactoryFriendly is OwnableUpgradeable { function setUp(bytes memory initializeParams) public virtual; }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity >=0.7.0 <0.9.0; import "@gnosis.pm/safe-contracts/contracts/common/Enum.sol"; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import "@gnosis.pm/safe-contracts/contracts/interfaces/IERC165.sol"; import "./BaseGuard.sol"; /// @title Guardable - A contract that manages fallback calls made to this contract contract Guardable is OwnableUpgradeable { event ChangedGuard(address guard); address public guard; /// @dev Set a guard that checks transactions before execution /// @param _guard The address of the guard to be used or the 0 address to disable the guard function setGuard(address _guard) external onlyOwner { if (_guard != address(0)) { require( BaseGuard(_guard).supportsInterface(type(IGuard).interfaceId), "Guard does not implement IERC165" ); } guard = _guard; emit ChangedGuard(guard); } function getGuard() external view returns (address _guard) { return guard; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import "../proxy/utils/Initializable.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal initializer { __Context_init_unchained(); __Ownable_init_unchained(); } function __Ownable_init_unchained() internal initializer { _setOwner(_msgSender()); } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { require(owner() == _msgSender(), "Ownable: caller is not the owner"); _; } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _setOwner(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _setOwner(newOwner); } function _setOwner(address newOwner) private { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } uint256[49] private __gap; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; import "../proxy/utils/Initializable.sol"; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract ContextUpgradeable is Initializable { function __Context_init() internal initializer { __Context_init_unchained(); } function __Context_init_unchained() internal initializer { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } uint256[50] private __gap; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since a proxied contract can't have a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. */ bool private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Modifier to protect an initializer function from being invoked twice. */ modifier initializer() { require(_initializing || !_initialized, "Initializable: contract is already initialized"); bool isTopLevelCall = !_initializing; if (isTopLevelCall) { _initializing = true; _initialized = true; } _; if (isTopLevelCall) { _initializing = false; } } }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity >=0.7.0 <0.9.0; /// @notice More details at https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/utils/introspection/IERC165.sol interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity >=0.7.0 <0.9.0; import "@gnosis.pm/safe-contracts/contracts/common/Enum.sol"; import "@gnosis.pm/safe-contracts/contracts/interfaces/IERC165.sol"; import "../interfaces/IGuard.sol"; abstract contract BaseGuard is IERC165 { function supportsInterface(bytes4 interfaceId) external pure override returns (bool) { return interfaceId == type(IGuard).interfaceId || // 0xe6d7a83a interfaceId == type(IERC165).interfaceId; // 0x01ffc9a7 } /// Module transactions only use the first four parameters: to, value, data, and operation. /// Module.sol hardcodes the remaining parameters as 0 since they are not used for module transactions. /// This interface is used to maintain compatibilty with Gnosis Safe transaction guards. function checkTransaction( address to, uint256 value, bytes memory data, Enum.Operation operation, uint256 safeTxGas, uint256 baseGas, uint256 gasPrice, address gasToken, address payable refundReceiver, bytes memory signatures, address msgSender ) external virtual; function checkAfterExecution(bytes32 txHash, bool success) external virtual; }
// SPDX-License-Identifier: LGPL-3.0-only pragma solidity >=0.7.0 <0.9.0; import "@gnosis.pm/safe-contracts/contracts/common/Enum.sol"; interface IGuard { function checkTransaction( address to, uint256 value, bytes memory data, Enum.Operation operation, uint256 safeTxGas, uint256 baseGas, uint256 gasPrice, address gasToken, address payable refundReceiver, bytes memory signatures, address msgSender ) external; function checkAfterExecution(bytes32 txHash, bool success) external; }
{ "optimizer": { "enabled": false, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "metadata": { "useLiteralContent": true }, "libraries": { "contracts/Permissions.sol": { "Permissions": "0x543d1de69b25420685ef723842d0087d9b731b06" } } }
[{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address","name":"_avatar","type":"address"},{"internalType":"address","name":"_target","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[],"name":"ArraysDifferentLength","type":"error"},{"inputs":[],"name":"ModuleTransactionFailed","type":"error"},{"inputs":[],"name":"NoMembership","type":"error"},{"inputs":[],"name":"SetUpModulesAlreadyCalled","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"module","type":"address"},{"indexed":false,"internalType":"uint16[]","name":"roles","type":"uint16[]"},{"indexed":false,"internalType":"bool[]","name":"memberOf","type":"bool[]"}],"name":"AssignRoles","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousAvatar","type":"address"},{"indexed":true,"internalType":"address","name":"newAvatar","type":"address"}],"name":"AvatarSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"guard","type":"address"}],"name":"ChangedGuard","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"module","type":"address"}],"name":"DisabledModule","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"module","type":"address"}],"name":"EnabledModule","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"initiator","type":"address"},{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"avatar","type":"address"},{"indexed":false,"internalType":"address","name":"target","type":"address"}],"name":"RolesModSetup","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"module","type":"address"},{"indexed":false,"internalType":"uint16","name":"defaultRole","type":"uint16"}],"name":"SetDefaultRole","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"multisendAddress","type":"address"}],"name":"SetMultisendAddress","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousTarget","type":"address"},{"indexed":true,"internalType":"address","name":"newTarget","type":"address"}],"name":"TargetSet","type":"event"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"enum ExecutionOptions","name":"options","type":"uint8"}],"name":"allowTarget","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"module","type":"address"},{"internalType":"uint16[]","name":"_roles","type":"uint16[]"},{"internalType":"bool[]","name":"memberOf","type":"bool[]"}],"name":"assignRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"avatar","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"defaultRoles","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"prevModule","type":"address"},{"internalType":"address","name":"module","type":"address"}],"name":"disableModule","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"module","type":"address"}],"name":"enableModule","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum Enum.Operation","name":"operation","type":"uint8"}],"name":"execTransactionFromModule","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum Enum.Operation","name":"operation","type":"uint8"}],"name":"execTransactionFromModuleReturnData","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"bytes","name":"","type":"bytes"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum Enum.Operation","name":"operation","type":"uint8"},{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"bool","name":"shouldRevert","type":"bool"}],"name":"execTransactionWithRole","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"enum Enum.Operation","name":"operation","type":"uint8"},{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"bool","name":"shouldRevert","type":"bool"}],"name":"execTransactionWithRoleReturnData","outputs":[{"internalType":"bool","name":"success","type":"bool"},{"internalType":"bytes","name":"returnData","type":"bytes"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getGuard","outputs":[{"internalType":"address","name":"_guard","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"start","type":"address"},{"internalType":"uint256","name":"pageSize","type":"uint256"}],"name":"getModulesPaginated","outputs":[{"internalType":"address[]","name":"array","type":"address[]"},{"internalType":"address","name":"next","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"guard","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_module","type":"address"}],"name":"isModuleEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"multisend","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"}],"name":"revokeTarget","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"enum ExecutionOptions","name":"options","type":"uint8"}],"name":"scopeAllowFunction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"bool[]","name":"isParamScoped","type":"bool[]"},{"internalType":"enum ParameterType[]","name":"paramType","type":"uint8[]"},{"internalType":"enum Comparison[]","name":"paramComp","type":"uint8[]"},{"internalType":"bytes[]","name":"compValue","type":"bytes[]"},{"internalType":"enum ExecutionOptions","name":"options","type":"uint8"}],"name":"scopeFunction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"enum ExecutionOptions","name":"options","type":"uint8"}],"name":"scopeFunctionExecutionOptions","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"uint256","name":"paramIndex","type":"uint256"},{"internalType":"enum ParameterType","name":"paramType","type":"uint8"},{"internalType":"enum Comparison","name":"paramComp","type":"uint8"},{"internalType":"bytes","name":"compValue","type":"bytes"}],"name":"scopeParameter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"uint256","name":"paramIndex","type":"uint256"},{"internalType":"enum ParameterType","name":"paramType","type":"uint8"},{"internalType":"bytes[]","name":"compValues","type":"bytes[]"}],"name":"scopeParameterAsOneOf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"}],"name":"scopeRevokeFunction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"}],"name":"scopeTarget","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_avatar","type":"address"}],"name":"setAvatar","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"module","type":"address"},{"internalType":"uint16","name":"role","type":"uint16"}],"name":"setDefaultRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_guard","type":"address"}],"name":"setGuard","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_multisend","type":"address"}],"name":"setMultisend","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_target","type":"address"}],"name":"setTarget","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes","name":"initParams","type":"bytes"}],"name":"setUp","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"target","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"role","type":"uint16"},{"internalType":"address","name":"targetAddress","type":"address"},{"internalType":"bytes4","name":"functionSig","type":"bytes4"},{"internalType":"uint8","name":"paramIndex","type":"uint8"}],"name":"unscopeParameter","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.