Upgrading smart contracts is a challenging task because they are immutable once deployed on a blockchain. This makes fixing bugs or adding new features difficult. However, using proxy smart contracts provides a way to overcome this limitation
A proxy contract is a smart contract that forwards function calls to another contract, known as the implementation contract. The proxy contract holds the state, while the implementation contract contains the logic. When you want to upgrade the contract, you deploy a new implementation and update the proxy to point to this new address. This way, you can change the contract’s behavior without affecting its address. There are several Proxy Smart Contract Patterns implementations exist.
User ---- tx ---> Proxy -X--------> Implementation_v0
|
---X--------> Implementation_v1
|
------------> Implementation_v2
Among the Proxy Smart Contract Patterns, the UUPS proxy pattern is the most suitable for our needs. It is not only straightforward to implement but also it provides flexability to update the upgrade logic in the future. The OpenZeppelin library can be used for this implementation. Conveniently, it offers seamless integration with the Hardhat library, simplifying our testing process. To implement this proxy using OpenZeppelin, it only need to inherit our contract from the UUPSUpgradeable contract. This contract provides an abstract method, _authorizeUpgrade, where we can define our own upgrade permission mechanism. Additionally, it has a built-in security feature to prevent any upgrades to non-UUPS compliant implementations. This is essential as it avoids unintentional upgrades to an implementation contract lacking the requisite upgrade mechanisms, which would otherwise permanently lock the proxy's upgradeability.
This smart contract is designed to manage the approval process for upgrading a contract. It is specifically built to work with OpenZeppelin's UUPSUpgradeable contracts. When a new implementation receives approval from more than 60 percent of the accounts with trustee role, the contract automatically upgrades the current implementation to the newly approved on.
- Proposals collection:
- Description: Double mapping holding the proposed upgrades and information about their approvals, can be accessed by combination of proxy and implementation addresses. The key relationship can be visualized as:
proxy address -> implementation address -> upgrade proposal
. - Format:
mapping(address => mapping(address => UpgradeProposal)) private upgradeProposals; struct UpgradeProposal { mapping (address => bool) approvals; uint approvalsCount; address author; uint256 created; }
- Description: Double mapping holding the proposed upgrades and information about their approvals, can be accessed by combination of proxy and implementation addresses. The key relationship can be visualized as:
-
Method:
ensureSufficientApprovals
- Description: This transaction ensures that an implementation upgrade has received sufficient approvals. At least 60% of users with the trustee role should approve before proceeding. If approvals are insufficient, the transaction will be reverted with a
InsufficientApprovals
error. It can be invoked within the _authorizeUpgrade method of the UUPSUpgradeable contract. - Restrictions: None.
- Format:
UpgradeController.ensureSufficientApprovals(address proxy, address implementation)
- Raised Event: None
- Description: This transaction ensures that an implementation upgrade has received sufficient approvals. At least 60% of users with the trustee role should approve before proceeding. If approvals are insufficient, the transaction will be reverted with a
-
Method:
propose
- Description: Transaction to propose an upgrade to a specified contract implementation.
- Restrictions:
- Sender must have TRUSTEE role assigned.
- Implementation must be UUPSUpgradeable.
- The same implementation upgrade can not be proposed more than once.
- Format
UpgradeController.propose(address proxy, address implementation)
- Example:
UpgradeController.propose( "0x0000000000000000000000000000000000004444" "0xe5414e3cf982222df96453cd910395a5c62a3b3d" )
- Raised Event: UpgradeProposed(proxy, implementation, sender)
-
Method:
approve
- Description: Transaction to approve an upgrade to a specified contract implementation. Once over 60 percent of approvals are received, this function upgrades implementation.
- Restrictions:
- Sender must have TRUSTEE role assigned.
- The approved implementation must have been previously proposed.
- An account can only approve each implementation upgrade once.
- Format
UpgradeController.approve(address proxy, address implementation)
- Example:
UpgradeController.approve( "0x0000000000000000000000000000000000004444" "0xe5414e3cf982222df96453cd910395a5c62a3b3d" )
- Raised Event: UpgradeApproved(proxy, implementation, sender), Upgraded(implementation)
// SPDX-License-Identifier: Apache-2.0
pragma solidity ^0.8.20;
import { Initializable } from "@openzeppelin/contracts/proxy/utils/Initializable.sol";
import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy";
contract UpgradableContract is UUPSUpgradable, Initializable {
UpgradeControlInterface _upgradeControl;
function initialize(address upgradeControlAddress) public reinitializer(1) {
_upgradeControl = UpgradeControlInterface(upgradeControlAddress);
}
function _authorizeUpgrade(address newImplementation) internal override {
_upgradeControl.ensureSufficientApprovals(address(this), newImplementation);
}
}
Alternatively, you can extend from the ControlledUpgreadable
contract, which encapsulates common boilerplate code:
// SPDX-License-Identifier: Apache-2.0
pragma solidity ^0.8.20;
import { UpgradeControlInterface } from "contracts/upgrade/ControlledUpgreadable.sol";
contract UpgradableContract is ControlledUpgreadable {
function initialize(address upgradeControlAddress) public reinitializer(1) {
// This method must be called on initialization to set the upgrade control,
// allowing the upgradeable contract to verify upgrade approvement.
_initializeUpgradeControl(upgradeControlAddress)
}
}
Transparent proxy pattern (EIP-1538)
In the transparent proxy pattern used for upgrade smart contracts, the implementation of updates resides on the proxy side. The proxy utilizes its own memory to store both the state of the contract and the address of the current implementation. The proxy implemets a fallback function that uses delegatecall to invoke the appropriate function in the linked implementation. Usually, only the admin or owner of this contract has the authority to update the implementation. A notable challenge with this pattern is the potential for function clashes. This occurs when the 4-byte hash of a function signature in the implementation matches that of a function in the proxy, leading to unintentional behavior and potential vulnerabilities. One common solution to mitigate this problem is to ensure that only the contract administrator has the rights to call proxy-specific methods, while other accounts are restricted to calling methods from the implementation.
Pros: Relatively straidforward to implement Cons: Deployment can be gas inefficent
UUPS proxy pattern (EIP-1822)
The UUPS proxy pattern is similar to the transparent proxy pattern. However, a distinct difference lies in where the update logic is implemented. In the UUPS pattern, the update logic is placed within the implementation contract, allowing for the possibility to modify this logic or even remove it entirely in the future. This introduces potential risks. If a bug is present in a new implementation, the contract update logic can be broken. While it's advisable to implement safeguards against unintentional breakdown of the update functionality, it's essential to note that these can't guard against intentional breakdowns.
Pros: Offers the options to update or eliminate upgrade logic. Cons: Upgrade should be executed carefally as it can break upgrage logic.
Beacon proxy is a proxy pattern which is a separate contract that holds the logic address for one or more proxy contracts. In this setup, the proxy contract doesn’t store the address of the logic contract directly. Instead, it points to a beacon contract, which in turn points to the logic contract. This allows multiple proxy contracts to share a single logic contract through a common beacon, making it easier to manage upgrades for a group of proxy contracts.
Pros: Highly efficient for deploying contracts with similar logic and reduces costs by using one implementation and beacon proxy. Cons: Complex to implement and maintain
Diamond proxy pattern (EIP-2535)
Diamond proxy pattern introduces the concept of "Diamonds" which are a more modular approach to smart contract upgrades. A Diamond is a contract that delegates calls to multiple function implementations, known as “facets.” Facets can be added, replaced, or removed, allowing for more flexible and modular upgrades. This standard also includes a way to query which facets are currently active.
Pros: Helps in addressing smart contract size limitations and functionality can be upgraded incrementally. Cons: Complex to implement and maintain