Hay 2 contratos A
e importaciones , donde hay una biblioteca que B
se puede actualizar. Para lograr esto, intenté reemplazarlo con un contrato de proxy y un contrato de delegado.A
B
B
B
Ahora tenemos contratos Foo
, que importa contrato de proxy Bar
, apuntando a contrato delegado ZeroDelegate
.
foo.sol
pragma solidity ^0.4.18;
import './Bar.sol';
contract Foo {
uint storageData;
Bar bar;
address barContractAddress;
constructor(address _barContractAddress) public {
barContractAddress = _barContractAddress;
}
function set(uint x) public {
storageData = x;
}
function get() view public returns (uint) {
return storageData;
}
function baz() public returns (uint) {
bar = Bar(barContractAddress);
storageData = bar.baz(storageData);
}
}
bar.sol
pragma solidity ^0.4.18;
import './Proxy.sol';
contract Bar is Proxy {
function baz(uint x) public returns (uint) {
return x * x;
}
}
ZeroDelegate.sol
pragma solidity ^0.4.18;
contract ZeroDelegate {
function baz(uint x) public returns (uint) {
return x * 0;
}
}
Proxy.sol
pragma solidity ^0.4.18;
import "zeppelin-solidity/contracts/ownership/Ownable.sol";
contract Proxy is Ownable {
event Upgraded(address indexed implementation);
address internal _implementation;
function implementation() public view returns (address) {
return _implementation;
}
function upgradeTo(address impl) public onlyOwner {
require(_implementation != impl);
_implementation = impl;
emit Upgraded(impl);
}
function () payable public {
address _impl = implementation();
require(_impl != address(0));
bytes memory data = msg.data;
assembly {
let result := delegatecall(gas, _impl, add(data, 0x20), mload(data), 0, 0)
let size := returndatasize
let ptr := mload(0x40)
returndatacopy(ptr, 0, size)
switch result
case 0 { revert(ptr, size) }
default { return(ptr, size) }
}
}
}
Ahora primero implementamos los contratos Foo
, Bar
, ZeroDelegate
.
bar = await Bar.new()
foo = await Foo.new(bar.address)
zeroDelegate = await ZeroDelegate.new()
Y foo.baz()
cuadra el número 2 a 4.
x = await foo.baz()
console.log(x.toNumber()) // 4
A continuación, actualizamos el Bar
contrato a ZeroDelegate
, pero foo.baz()
igual cuadramos el número 4 a 16
await bar.upgradeTo(zeroDelegate.address)
bar = _.extend(bar, ZeroDelegate.at(bar.address))
await foo.baz()
x = await foo.get()
console.log(x.toNumber()) // 16, but expects 0
Sin embargo, si tuviéramos que volver a implementar Foo
el contrato, utiliza el actualizado Bar
. ¿Por qué es esto y cómo podemos permitir Foo
que usemos las Bar
funciones actualizadas sin tener que volver a implementar Foo
, lo que anula el propósito de usar un contrato actualizable?
foo = await Foo.new(bar.address)
await foo.baz()
x = await foo.get()
console.log(x.toNumber()) // 0
Debe hacer que su función de respaldo se llame baz(), luego trate a Bar.sol de la misma manera que trata a ZeroDelegate (como un contrato externo no heredado, tal vez en realidad una biblioteca). La llamada delegada se enviará a ese contrato externo (cualquiera que sea el _imp que esté apuntando actualmente) y utilizará la lógica dentro de ese contrato. Cuando actualice _imp a la dirección de Bar o ZeroDelegate para cambiar su comportamiento.
Aquí hay un ejemplo de trabajo con una función TokenURI actualizable:
https://github.com/clovers-network/clovers-contracts/blob/master/contracts/Clovers.sol
que apunta a:
https://github.com/clovers-network/ tréboles-contratos/blob/master/contratos/CloversMetadata.sol