title | tags | |||
---|---|---|---|---|
49. Proxy Universal Atualizável |
|
Recentemente, estou revisando Solidity para consolidar alguns detalhes e escrever um "WTF Solidity: Guia Básico" para iniciantes (programadores experientes podem procurar outros tutoriais). Serão lançadas de 1 a 3 aulas por semana.
Twitter: @0xAA_Science
Comunidade: Discord | Grupo do WeChat | Site oficial wtf.academy
Todo o código e tutoriais estão disponíveis no GitHub: github.com/AmazingAng/WTFSolidity
Nesta aula, vamos apresentar outra solução para o problema de conflito de seletores (Selector Clash) em contratos de proxy: o Proxy Universal Atualizável (UUPS, universal upgradeable proxy standard). O código do tutorial é simplificado a partir do contrato UUPSUpgradeable da OpenZeppelin e não deve ser usado em produção.
Na aula anterior, aprendemos sobre o "conflito de seletores" (Selector Clash), que ocorre quando um contrato possui duas funções com o mesmo seletor, o que pode causar sérios problemas. Como uma alternativa ao proxy transparente, o UUPS também pode resolver esse problema.
O Proxy Universal Atualizável (UUPS, universal upgradeable proxy standard) coloca a função de atualização no contrato lógico. Dessa forma, se houver outras funções que entrem em conflito com a função de atualização, um erro de compilação será gerado.
A tabela a seguir resume as diferenças entre contratos atualizáveis comuns, proxies transparentes e UUPS:
Primeiro, vamos revisar a Aula 23 do WTF Solidity: Delegatecall. Se o usuário A fizer um delegatecall
para o contrato C (contrato lógico) por meio do contrato B (contrato de proxy), o contexto ainda será o do contrato B e o msg.sender
será o usuário A, não o contrato B. Portanto, o contrato UUPS pode colocar a função de atualização no contrato lógico e verificar se o chamador é o administrador.
O contrato de proxy UUPS se parece com um contrato de proxy não atualizável e é muito simples, porque a função de atualização está no contrato lógico. Ele contém três variáveis:
implementation
: endereço do contrato lógico.admin
: endereço do administrador.words
: uma string que pode ser alterada por meio de funções do contrato lógico.
Ele contém duas funções:
- Construtor: inicializa o endereço do administrador e do contrato lógico.
fallback()
: função de fallback que delega a chamada para o contrato lógico.
contract UUPSProxy {
address public implementation; // endereço do contrato lógico
address public admin; // endereço do administrador
string public words; // uma string que pode ser alterada por meio de funções do contrato lógico
// Construtor: inicializa o endereço do administrador e do contrato lógico
constructor(address _implementation){
admin = msg.sender;
implementation = _implementation;
}
// fallback: delega a chamada para o contrato lógico
fallback() external payable {
(bool success, bytes memory data) = implementation.delegatecall(msg.data);
}
}
O contrato lógico UUPS é diferente do contrato apresentado na Aula 47 porque agora possui uma função de atualização. O contrato lógico UUPS contém três variáveis de estado, que são as mesmas do contrato de proxy para evitar conflitos de slots. Ele contém duas funções:
upgrade()
: função de atualização que altera o endereço do contrato lógicoimplementation
e só pode ser chamada peloadmin
.foo()
: a versão antiga do contrato UUPS altera o valor dewords
para"old"
, enquanto a nova versão altera para"new"
.
// Contrato lógico UUPS (função de atualização no contrato lógico)
contract UUPS1{
// Variáveis de estado que são as mesmas do contrato de proxy para evitar conflitos de slots
address public implementation;
address public admin;
string public words; // uma string que pode ser alterada por meio de funções do contrato lógico
// Altera a variável de estado do contrato de proxy, seletor: 0xc2985578
function foo() public{
words = "old";
}
// Função de atualização que altera o endereço do contrato lógico e só pode ser chamada pelo admin, seletor: 0x0900f010
// No UUPS, o contrato lógico deve conter a função de atualização, caso contrário, não poderá ser atualizado novamente.
function upgrade(address newImplementation) external {
require(msg.sender == admin);
implementation = newImplementation;
}
}
// Nova versão do contrato lógico UUPS
contract UUPS2{
// Variáveis de estado que são as mesmas do contrato de proxy para evitar conflitos de slots
address public implementation;
address public admin;
string public words; // uma string que pode ser alterada por meio de funções do contrato lógico
// Altera a variável de estado do contrato de proxy, seletor: 0xc2985578
function foo() public{
words = "new";
}
// Função de atualização que altera o endereço do contrato lógico e só pode ser chamada pelo admin, seletor: 0x0900f010
// No UUPS, o contrato lógico deve conter a função de atualização, caso contrário, não poderá ser atualizado novamente.
function upgrade(address newImplementation) external {
require(msg.sender == admin);
implementation = newImplementation;
}
}
- Implante as versões antigas e novas do contrato lógico UUPS,
UUPS1
eUUPS2
.
- Implante o contrato de proxy UUPS,
UUPSProxy
, e defina o endereço deimplementation
como o do contrato lógico antigo,UUPS1
.
- Usando o seletor
0xc2985578
, chame a funçãofoo()
do contrato lógico antigo,UUPS1
, no contrato de proxy para alterar o valor dewords
para"old"
.
- Usando um codificador ABI online, como o HashEx, obtenha a codificação binária e chame a função de atualização
upgrade()
para definir o endereço deimplementation
como o do contrato lógico novo,UUPS2
.
- Usando o seletor
0xc2985578
, chame a funçãofoo()
do contrato lógico novo,UUPS2
, no contrato de proxy para alterar o valor dewords
para"new"
.
Nesta aula, apresentamos outra solução para o problema de "conflito de seletores" em contratos de proxy: o UUPS. Ao contrário do proxy transparente, o UUPS coloca a função de atualização no contrato lógico, o que impede que ocorra um conflito de seletores durante a compilação. Comparado ao proxy transparente, o UUPS consome menos gas, mas também é mais complexo.