Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Consider making Upgradeable CoreMetadataModule #99

Merged
merged 6 commits into from
May 22, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions contracts/lib/Errors.sol
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,11 @@ library Errors {
/// @notice Provided calldata is invalid.
error IPAccount__InvalidCalldata();

////////////////////////////////////////////////////////////////////////////
// CoreMetadataModule //
////////////////////////////////////////////////////////////////////////////
error CoreMetadataModule__ZeroAccessManager();

////////////////////////////////////////////////////////////////////////////
// IP Account Storage //
////////////////////////////////////////////////////////////////////////////
Expand Down
35 changes: 31 additions & 4 deletions contracts/modules/metadata/CoreMetadataModule.sol
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@ pragma solidity ^0.8.23;

import { IERC165 } from "@openzeppelin/contracts/utils/introspection/IERC165.sol";
import { IERC721Metadata } from "@openzeppelin/contracts/token/ERC721/extensions/IERC721Metadata.sol";
import { UUPSUpgradeable } from "@openzeppelin/contracts-upgradeable/proxy/utils/UUPSUpgradeable.sol";
// solhint-disable-next-line max-line-length
import { AccessManagedUpgradeable } from "@openzeppelin/contracts-upgradeable/access/manager/AccessManagedUpgradeable.sol";
import { IIPAccount } from "../../interfaces/IIPAccount.sol";
import { AccessControlled } from "../../access/AccessControlled.sol";
import { BaseModule } from "../BaseModule.sol";
Expand All @@ -17,11 +20,15 @@ import { ICoreMetadataModule } from "../../interfaces/modules/metadata/ICoreMeta
/// It implements the ICoreMetadataModule interface.
/// The metadata can be set and updated by the owner of the IP asset.
/// The metadata can be frozen to prevent further changes.
contract CoreMetadataModule is BaseModule, AccessControlled, ICoreMetadataModule {
contract CoreMetadataModule is
BaseModule,
AccessControlled,
ICoreMetadataModule,
AccessManagedUpgradeable,
UUPSUpgradeable
{
using IPAccountStorageOps for IIPAccount;

string public override name = CORE_METADATA_MODULE_KEY;

/// @notice Modifier to ensure that metadata can only be changed when mutable.
modifier onlyMutable(address ipId) {
if (IIPAccount(payable(ipId)).getBool("IMMUTABLE")) {
Expand All @@ -33,10 +40,26 @@ contract CoreMetadataModule is BaseModule, AccessControlled, ICoreMetadataModule
/// @notice Creates a new CoreMetadataModule instance.
/// @param accessController The address of the AccessController contract.
/// @param ipAccountRegistry The address of the IPAccountRegistry contract.
/// @custom:oz-upgrades-unsafe-allow constructor
constructor(
address accessController,
address ipAccountRegistry
) AccessControlled(accessController, ipAccountRegistry) {}
) AccessControlled(accessController, ipAccountRegistry) {
_disableInitializers();
}

/// @notice Initializes the CoreMetadataModule contract.
function initialize(address accessManager) public initializer {
if (accessManager == address(0)) {
revert Errors.CoreMetadataModule__ZeroAccessManager();
}
__AccessManaged_init(accessManager);
Ramarti marked this conversation as resolved.
Show resolved Hide resolved
__UUPSUpgradeable_init();
}

function name() external pure override returns (string memory) {
return CORE_METADATA_MODULE_KEY;
}

/// @notice Update the nftTokenURI for an IP asset,
/// by retrieve the latest TokenURI from IP NFT to which the IP Asset bound.
Expand Down Expand Up @@ -113,4 +136,8 @@ contract CoreMetadataModule is BaseModule, AccessControlled, ICoreMetadataModule
IIPAccount(payable(ipId)).setBytes32("METADATA_HASH", metadataHash);
emit MetadataURISet(ipId, metadataURI, metadataHash);
}

/// @dev Hook to authorize the upgrade according to UUPSUpgradeable
/// @param newImplementation The address of the new implementation
function _authorizeUpgrade(address newImplementation) internal override restricted {}
}
16 changes: 11 additions & 5 deletions script/foundry/utils/DeployHelper.sol
Original file line number Diff line number Diff line change
Expand Up @@ -500,15 +500,20 @@ contract DeployHelper is Script, BroadcastManager, JsonDeploymentHandler, Storag
_postdeploy("IpRoyaltyVaultBeacon", address(ipRoyaltyVaultBeacon));

_predeploy("CoreMetadataModule");
impl = address(new CoreMetadataModule(address(accessController), address(ipAssetRegistry)));
coreMetadataModule = CoreMetadataModule(
create3Deployer.deploy(
TestProxyHelper.deployUUPSProxy(
create3Deployer,
_getSalt(type(CoreMetadataModule).name),
abi.encodePacked(
type(CoreMetadataModule).creationCode,
abi.encode(address(accessController), address(ipAssetRegistry))
)
impl,
abi.encodeCall(CoreMetadataModule.initialize, address(protocolAccessManager))
)
Ramarti marked this conversation as resolved.
Show resolved Hide resolved
);
require(
_getDeployedAddress(type(CoreMetadataModule).name) == address(coreMetadataModule),
"Deploy: Core Metadata Module Address Mismatch"
);
require(_loadProxyImpl(address(coreMetadataModule)) == impl, "CoreMetadataModule Proxy Implementation Mismatch");
_postdeploy("CoreMetadataModule", address(coreMetadataModule));

_predeploy("CoreMetadataViewModule");
Expand Down Expand Up @@ -608,6 +613,7 @@ contract DeployHelper is Script, BroadcastManager, JsonDeploymentHandler, Storag
protocolAccessManager.setTargetFunctionRole(address(licenseRegistry), selectors, ProtocolAdmin.UPGRADER_ROLE);
protocolAccessManager.setTargetFunctionRole(address(moduleRegistry), selectors, ProtocolAdmin.UPGRADER_ROLE);
protocolAccessManager.setTargetFunctionRole(address(ipAssetRegistry), selectors, ProtocolAdmin.UPGRADER_ROLE);
protocolAccessManager.setTargetFunctionRole(address(coreMetadataModule), selectors, ProtocolAdmin.UPGRADER_ROLE);

// Royalty and Upgrade Beacon
// Owner of the beacon is the RoyaltyPolicyLAP
Expand Down
15 changes: 15 additions & 0 deletions test/foundry/upgrades/Upgrades.t.sol
Original file line number Diff line number Diff line change
Expand Up @@ -269,5 +269,20 @@ contract UpgradesTest is BaseTest {
),
ProtocolAdmin.UPGRADER_ROLE
);

(immediate, delay) = protocolAccessManager.canCall(
multisig,
address(coreMetadataModule),
UUPSUpgradeable.upgradeToAndCall.selector
);
assertFalse(immediate);
assertEq(delay, execDelay);
assertEq(
protocolAccessManager.getTargetFunctionRole(
address(coreMetadataModule),
UUPSUpgradeable.upgradeToAndCall.selector
),
ProtocolAdmin.UPGRADER_ROLE
);
}
}
Loading