From 82fe13aeb8c27445f438d6a0a725e9849c3f5a79 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Alberto=20Cuesta=20Ca=C3=B1ada?= Date: Mon, 13 Sep 2021 07:34:47 +0100 Subject: [PATCH] fix: require that `witch.setIlk` is called before `wand.makeIlk` --- contracts/Wand.sol | 16 +++++++++++----- contracts/Witch.sol | 2 ++ test/081_witch.ts | 1 + test/shared/fixtures.ts | 1 + 4 files changed, 15 insertions(+), 5 deletions(-) diff --git a/contracts/Wand.sol b/contracts/Wand.sol index dd96933e9..e29a98163 100644 --- a/contracts/Wand.sol +++ b/contracts/Wand.sol @@ -2,6 +2,7 @@ pragma solidity 0.8.6; import "@yield-protocol/vault-interfaces/ICauldronGov.sol"; import "@yield-protocol/vault-interfaces/ILadleGov.sol"; +// import "@yield-protocol/vault-interfaces/IWitchGov.sol"; import "@yield-protocol/vault-interfaces/IMultiOracleGov.sol"; import "@yield-protocol/vault-interfaces/IJoinFactory.sol"; import "@yield-protocol/vault-interfaces/IJoin.sol"; @@ -12,6 +13,9 @@ import "@yield-protocol/yieldspace-interfaces/IPoolFactory.sol"; import "@yield-protocol/utils-v2/contracts/access/AccessControl.sol"; import "./constants/Constants.sol"; +interface IWitchGov { + function ilks(bytes6) external view returns(bool, uint32, uint64, uint128); +} /// @dev Ladle orchestrates contract calls throughout the Yield Protocol v2 into useful and efficient governance features. contract Wand is AccessControl, Constants { @@ -25,7 +29,7 @@ contract Wand is AccessControl, Constants { ICauldronGov public cauldron; ILadleGov public ladle; - address public witch; + IWitchGov public witch; IPoolFactory public poolFactory; IJoinFactory public joinFactory; IFYTokenFactory public fyTokenFactory; @@ -33,7 +37,7 @@ contract Wand is AccessControl, Constants { constructor ( ICauldronGov cauldron_, ILadleGov ladle_, - address witch_, + IWitchGov witch_, IPoolFactory poolFactory_, IJoinFactory joinFactory_, IFYTokenFactory fyTokenFactory_ @@ -50,7 +54,7 @@ contract Wand is AccessControl, Constants { function point(bytes32 param, address value) external auth { if (param == "cauldron") cauldron = ICauldronGov(value); else if (param == "ladle") ladle = ILadleGov(value); - else if (param == "witch") witch = value; + else if (param == "witch") witch = IWitchGov(value); else if (param == "poolFactory") poolFactory = IPoolFactory(value); else if (param == "joinFactory") joinFactory = IJoinFactory(value); else if (param == "fyTokenFactory") fyTokenFactory = IFYTokenFactory(value); @@ -88,18 +92,20 @@ contract Wand is AccessControl, Constants { cauldron.setLendingOracle(assetId, IOracle(address(oracle))); AccessControl baseJoin = AccessControl(address(ladle.joins(assetId))); - baseJoin.grantRole(JOIN, witch); // Give the Witch permission to join base + baseJoin.grantRole(JOIN, address(witch)); // Give the Witch permission to join base } /// @dev Make an ilk asset out of a generic asset. /// @notice `oracle` must be able to deliver a value for baseId and ilkId function makeIlk(bytes6 baseId, bytes6 ilkId, IMultiOracleGov oracle, uint32 ratio, uint96 max, uint24 min, uint8 dec) external auth { require (address(oracle) != address(0), "Oracle required"); + (bool ilkInitialized,,,) = witch.ilks(ilkId); + require (ilkInitialized == true, "Initialize ilk in Witch"); cauldron.setSpotOracle(baseId, ilkId, IOracle(address(oracle)), ratio); cauldron.setDebtLimits(baseId, ilkId, max, min, dec); AccessControl ilkJoin = AccessControl(address(ladle.joins(ilkId))); - ilkJoin.grantRole(EXIT, witch); // Give the Witch permission to exit ilk + ilkJoin.grantRole(EXIT, address(witch)); // Give the Witch permission to exit ilk } /// @dev Add an existing series to the protocol, by deploying a FYToken, and registering it in the cauldron with the approved ilks diff --git a/contracts/Witch.sol b/contracts/Witch.sol index 2b0bf9491..ec5dc41e6 100644 --- a/contracts/Witch.sol +++ b/contracts/Witch.sol @@ -31,6 +31,7 @@ contract Witch is AccessControl() { } struct Ilk { + bool initialized; // Set to true if set, as we might want all parameters set to zero uint32 duration; // Time that auctions take to go to minimal price and stay there. uint64 initialOffer; // Proportion of collateral that is sold at auction start (1e18 = 100%) uint128 dust; // Minimum collateral that must be left when buying, unless buying all @@ -65,6 +66,7 @@ contract Witch is AccessControl() { function setIlk(bytes6 ilkId, uint32 duration, uint64 initialOffer, uint128 dust) external auth { require (initialOffer <= 1e18, "Only at or under 100%"); ilks[ilkId] = Ilk({ + initialized: true, duration: duration, initialOffer: initialOffer, dust: dust diff --git a/test/081_witch.ts b/test/081_witch.ts index 5665ffb36..4f314a19b 100644 --- a/test/081_witch.ts +++ b/test/081_witch.ts @@ -112,6 +112,7 @@ describe('Witch', function () { expect(await witch.setIlk(ilkId, 1, 2, 3)) .to.emit(witch, 'IlkSet') .withArgs(ilkId, 1, 2, 3) + expect((await witch.ilks(ilkId)).initialized).to.be.true expect((await witch.ilks(ilkId)).duration).to.equal(1) expect((await witch.ilks(ilkId)).initialOffer).to.equal(2) expect((await witch.ilks(ilkId)).dust).to.equal(3) diff --git a/test/shared/fixtures.ts b/test/shared/fixtures.ts index 6bce60610..df6841f5d 100644 --- a/test/shared/fixtures.ts +++ b/test/shared/fixtures.ts @@ -364,6 +364,7 @@ export class YieldEnvironment { const base = assets.get(baseId) as ERC20Mock const ilk = assets.get(ilkId) as ERC20Mock await spotOracle.setSource(baseId, base.address, ilkId, ilk.address, spotSource.address) + await witch.setIlk(ilkId, 4 * 60 * 60, WAD.div(2), 0) await wand.makeIlk(baseId, ilkId, spotOracle.address, ratio, 1000000, 1, await base.decimals()) oracles.set(ilkId, (spotOracle as unknown) as OracleMock) }