forked from AmazingAng/WTF-Solidity
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTransparentProxy.sol
65 lines (55 loc) · 2.29 KB
/
TransparentProxy.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
// SPDX-License-Identifier: MIT
// wtf.academy
pragma solidity ^0.8.21;
// Exemplo de conflito de seletor
// Após remover os comentários, o contrato não será compilado, pois as duas funções têm o mesmo seletor.
contract Foo {
bytes4 public selector1 = bytes4(keccak256("burn(uint256)"));
bytes4 public selector2 = bytes4(keccak256("collate_propagate_storage(bytes16)"));
// function burn(uint256) external {}
// function collate_propagate_storage(bytes16) external {}
}
// Código de ensino para um contrato transparente e atualizável, não utilizar em produção.
contract TransparentProxy {
// Endereço do contrato lógico
// Administrador
// Strings, podem ser alterados por meio de funções de contrato lógico.
// Construtor, inicializa os endereços do admin e do contrato lógico
constructor(address _implementation){
admin = msg.sender;
implementation = _implementation;
}
// fallback function, delegates the call to the logic contract
// Não pode ser chamado pelo admin para evitar conflitos de seletor inesperados
fallback() external payable {
require(msg.sender != admin);
(bool success, bytes memory data) = implementation.delegatecall(msg.data);
}
// Função de atualização, altera o endereço do contrato lógico, só pode ser chamada pelo admin
function upgrade(address newImplementation) external {
if (msg.sender != admin) revert();
implementation = newImplementation;
}
}
// Contrato de lógica antigo
contract Logic1 {
// Variáveis de estado e contratos proxy são consistentes para evitar conflitos de slot
address public implementation;
address public admin;
// Strings, podem ser alterados por meio de funções de contrato lógico.
// Alterando a variável de estado no proxy, seletor: 0xc2985578
function foo() public{
words = "old";
}
}
// Novo contrato lógico
contract Logic2 {
// Variáveis de estado e contratos proxy são consistentes para evitar conflitos de slot
address public implementation;
address public admin;
// Strings, podem ser alterados por meio de funções de contrato lógico.
// Alterando a variável de estado no proxy, seletor: 0xc2985578
function foo() public{
words = "new";
}
}