I have recently been revising Solidity to consolidate the details, and am writing a "WTF Simplified Introduction to Solidity" for beginners to use (programming experts can find other tutorials), with weekly updates of 1-3 lectures.
Twitter: @0xAA_Science
Community: Discord|WeChat group|Official website wtf.academy
All code and tutorials are open source on GitHub: github.com/AmazingAng/WTF-Solidity
in this lesson, we'll introduce Upgradeable Contract, the sample contracts used for teaching are simplified from OpenZeppelin
contracts, they may pose security issues, DO NOT USE IN PRODUCTION.
Upgradeable Contract
If you understand proxy contracts, it is easy to understand upgradeable contracts. It is a proxy contract that can change the logic contract.
data:image/s3,"s3://crabby-images/287c9/287c9f5375851aca1789c48a4b06ff2be5fabfd3" alt="Upgradeable Contract Pattern"
Simple Implementation
Below we implement a simple upgradeable contract that includes 3 contracts: the proxy contract, the old logic contract, and the new logic contract.
Proxy Contract
This proxy contract is simpler than that in Lecture 46. We didn't use inline assembly in its fallback()
function, but only used implementation.delegatecall(msg.data);
. Therefore, the callback function does not return a value, but it is sufficient for teaching purposes.
It contains 3 variables:
implementation
: The logic contract address.admin
: Admin address.words
: String that can be changed through a function in the logic contract.
It contains 3
functions:
- Constructor: Initializes admin and logic contract addresses.
fallback()
: Callback function, delegates the call to the logic contract.upgrade()
: The upgrade function, changes the logic contract's address, and can only be called byadmin
.
Old Logic Contract
This logic contract includes 3
state variables and is consistent with the proxy contract to prevent slot conflicts. It only has one function foo()
, which changes the value of words
in the proxy contract to "old"
.
New Logic Contract
This logic contract contains 3
state variables, consistent with the proxy contract to prevent slot conflicts. It has only one function, foo()
, which changes the value of words
in the proxy contract to "new"
.
Implementation with Remix
-
Deploy the old and new logic contracts
Logic1
andLogic2
. -
Deploy the upgradeable contract
SimpleUpgrade
and set theimplementation
address to the address of the old logic contract. -
Use the selector
0xc2985578
to call thefoo()
function of the old logic contractLogic1
in the proxy contract and change the value ofwords
to"old"
. -
Call
upgrade()
to set theimplementation
address to the address of the new logic contractLogic2
. -
Use the selector
0xc2985578
to call thefoo()
function of the new logic contractLogic2
in the proxy contract and change the value ofwords
to"new"
.
Summary
In this lesson, we introduced a simple upgradeable contract. It is a proxy contract that can change the logic contract and add upgrade functionality to immutable smart contracts. However, this contract has a problem of selector conflict and poses security risks. Later, we will introduce the upgradeable contract standards that solve this vulnerability: Transparent Proxy and UUPS.