Skip to content

Commit 2e84093

Browse files
authoredNov 29, 2021
Use a minimal proxy for the curation shares ERC20 (#505)
* curation: use a minimal proxy to save gas when minting for first time * curation: use token utils for transfers * tests: remove unnecessary block advance in test * curation: add external function to set token master copy and avoid re-deploy the clone on minting reset * chore: add graph curation master copy token to deployment config
1 parent 0877142 commit 2e84093

13 files changed

+133
-62
lines changed
 

‎cli/commands/migrate.ts

+1
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@ let allContracts = [
2222
'Controller',
2323
'EpochManager',
2424
'GraphToken',
25+
'GraphCurationToken',
2526
'ServiceRegistry',
2627
'Curation',
2728
'GNS',

‎contracts/curation/Curation.sol

+46-40
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,13 @@
22

33
pragma solidity ^0.7.6;
44

5+
import "@openzeppelin/contracts/utils/Address.sol";
56
import "@openzeppelin/contracts/math/SafeMath.sol";
7+
import "@openzeppelin/contracts/proxy/Clones.sol";
68

79
import "../bancor/BancorFormula.sol";
810
import "../upgrades/GraphUpgradeable.sol";
11+
import "../utils/TokenUtils.sol";
912

1013
import "./CurationStorage.sol";
1114
import "./ICuration.sol";
@@ -23,7 +26,7 @@ import "./GraphCurationToken.sol";
2326
* Holders can burn GCS using this contract to get GRT tokens back according to the
2427
* bonding curve.
2528
*/
26-
contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
29+
contract Curation is CurationV1Storage, GraphUpgradeable {
2730
using SafeMath for uint256;
2831

2932
// 100% in parts per million
@@ -70,6 +73,7 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
7073
function initialize(
7174
address _controller,
7275
address _bondingCurve,
76+
address _curationTokenMaster,
7377
uint32 _defaultReserveRatio,
7478
uint32 _curationTaxPercentage,
7579
uint256 _minimumCurationDeposit
@@ -83,6 +87,7 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
8387
_setDefaultReserveRatio(_defaultReserveRatio);
8488
_setCurationTaxPercentage(_curationTaxPercentage);
8589
_setMinimumCurationDeposit(_minimumCurationDeposit);
90+
_setCurationTokenMaster(_curationTokenMaster);
8691
}
8792

8893
/**
@@ -154,10 +159,30 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
154159
"Curation tax percentage must be below or equal to MAX_PPM"
155160
);
156161

157-
_curationTaxPercentage = _percentage;
162+
curationTaxPercentage = _percentage;
158163
emit ParameterUpdated("curationTaxPercentage");
159164
}
160165

166+
/**
167+
* @dev Set the master copy to use as clones for the curation token.
168+
* @param _curationTokenMaster Address of implementation contract to use for curation tokens
169+
*/
170+
function setCurationTokenMaster(address _curationTokenMaster) external override onlyGovernor {
171+
_setCurationTokenMaster(_curationTokenMaster);
172+
}
173+
174+
/**
175+
* @dev Internal: Set the master copy to use as clones for the curation token.
176+
* @param _curationTokenMaster Address of implementation contract to use for curation tokens
177+
*/
178+
function _setCurationTokenMaster(address _curationTokenMaster) private {
179+
require(_curationTokenMaster != address(0), "Token master must be non-empty");
180+
require(Address.isContract(_curationTokenMaster), "Token master must be a contract");
181+
182+
curationTokenMaster = _curationTokenMaster;
183+
emit ParameterUpdated("curationTokenMaster");
184+
}
185+
161186
/**
162187
* @dev Assign Graph Tokens collected as curation fees to the curation pool reserve.
163188
* This function can only be called by the Staking contract and will do the bookeeping of
@@ -208,36 +233,27 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
208233

209234
// If it hasn't been curated before then initialize the curve
210235
if (!isCurated(_subgraphDeploymentID)) {
211-
// Initialize
212236
curationPool.reserveRatio = defaultReserveRatio;
213237

214238
// If no signal token for the pool - create one
215239
if (address(curationPool.gcs) == address(0)) {
216-
// TODO: Use a minimal proxy to reduce gas cost
217-
// https://github.com/graphprotocol/contracts/issues/405
218-
// --abarmat-- 20201113
219-
curationPool.gcs = IGraphCurationToken(
220-
address(new GraphCurationToken(address(this)))
221-
);
240+
// Use a minimal proxy to reduce gas cost
241+
IGraphCurationToken gcs = IGraphCurationToken(Clones.clone(curationTokenMaster));
242+
gcs.initialize(address(this));
243+
curationPool.gcs = gcs;
222244
}
223245
}
224246

225247
// Trigger update rewards calculation snapshot
226248
_updateRewards(_subgraphDeploymentID);
227249

228250
// Transfer tokens from the curator to this contract
229-
// This needs to happen after _updateRewards snapshot as that function
251+
// Burn the curation tax
252+
// NOTE: This needs to happen after _updateRewards snapshot as that function
230253
// is using balanceOf(curation)
231-
IGraphToken graphToken = graphToken();
232-
require(
233-
graphToken.transferFrom(curator, address(this), _tokensIn),
234-
"Cannot transfer tokens to deposit"
235-
);
236-
237-
// Burn withdrawal fees
238-
if (curationTax > 0) {
239-
graphToken.burn(curationTax);
240-
}
254+
IGraphToken _graphToken = graphToken();
255+
TokenUtils.pullTokens(_graphToken, curator, _tokensIn);
256+
TokenUtils.burnTokens(_graphToken, curationTax);
241257

242258
// Update curation pool
243259
curationPool.tokens = curationPool.tokens.add(_tokensIn.sub(curationTax));
@@ -284,13 +300,15 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
284300
curationPool.tokens = curationPool.tokens.sub(tokensOut);
285301
curationPool.gcs.burnFrom(curator, _signalIn);
286302

287-
// If all signal burnt delete the curation pool
303+
// If all signal burnt delete the curation pool except for the
304+
// curation token contract to avoid recreating it on a new mint
288305
if (getCurationPoolSignal(_subgraphDeploymentID) == 0) {
289-
delete pools[_subgraphDeploymentID];
306+
curationPool.tokens = 0;
307+
curationPool.reserveRatio = 0;
290308
}
291309

292310
// Return the tokens to the curator
293-
require(graphToken().transfer(curator, tokensOut), "Error sending curator tokens");
311+
TokenUtils.pushTokens(graphToken(), curator, tokensOut);
294312

295313
emit Burned(curator, _subgraphDeploymentID, tokensOut, _signalIn);
296314

@@ -318,10 +336,8 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
318336
override
319337
returns (uint256)
320338
{
321-
if (address(pools[_subgraphDeploymentID].gcs) == address(0)) {
322-
return 0;
323-
}
324-
return pools[_subgraphDeploymentID].gcs.balanceOf(_curator);
339+
IGraphCurationToken gcs = pools[_subgraphDeploymentID].gcs;
340+
return (address(gcs) == address(0)) ? 0 : gcs.balanceOf(_curator);
325341
}
326342

327343
/**
@@ -335,10 +351,8 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
335351
override
336352
returns (uint256)
337353
{
338-
if (address(pools[_subgraphDeploymentID].gcs) == address(0)) {
339-
return 0;
340-
}
341-
return pools[_subgraphDeploymentID].gcs.totalSupply();
354+
IGraphCurationToken gcs = pools[_subgraphDeploymentID].gcs;
355+
return (address(gcs) == address(0)) ? 0 : gcs.totalSupply();
342356
}
343357

344358
/**
@@ -355,14 +369,6 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
355369
return pools[_subgraphDeploymentID].tokens;
356370
}
357371

358-
/**
359-
* @dev Get curation tax percentage
360-
* @return Amount the curation tax percentage in PPM
361-
*/
362-
function curationTaxPercentage() external view override returns (uint32) {
363-
return _curationTaxPercentage;
364-
}
365-
366372
/**
367373
* @dev Calculate amount of signal that can be bought with tokens in a curation pool.
368374
* This function considers and excludes the deposit tax.
@@ -376,7 +382,7 @@ contract Curation is CurationV1Storage, GraphUpgradeable, ICuration {
376382
override
377383
returns (uint256, uint256)
378384
{
379-
uint256 curationTax = _tokensIn.mul(uint256(_curationTaxPercentage)).div(MAX_PPM);
385+
uint256 curationTax = _tokensIn.mul(uint256(curationTaxPercentage)).div(MAX_PPM);
380386
uint256 signalOut = _tokensToSignal(_subgraphDeploymentID, _tokensIn.sub(curationTax));
381387
return (signalOut, curationTax);
382388
}

‎contracts/curation/CurationStorage.sol

+16-5
Original file line numberDiff line numberDiff line change
@@ -2,28 +2,39 @@
22

33
pragma solidity ^0.7.6;
44

5-
import "./ICuration.sol";
65
import "../governance/Managed.sol";
76

8-
contract CurationV1Storage is Managed {
7+
abstract contract CurationV1Storage is Managed, ICuration {
8+
// -- Pool --
9+
10+
struct CurationPool {
11+
uint256 tokens; // GRT Tokens stored as reserves for the subgraph deployment
12+
uint32 reserveRatio; // Ratio for the bonding curve
13+
IGraphCurationToken gcs; // Curation token contract for this curation pool
14+
}
15+
916
// -- State --
1017

1118
// Tax charged when curator deposit funds
1219
// Parts per million. (Allows for 4 decimal points, 999,999 = 99.9999%)
13-
uint32 internal _curationTaxPercentage;
20+
uint32 public override curationTaxPercentage;
1421

1522
// Default reserve ratio to configure curator shares bonding curve
1623
// Parts per million. (Allows for 4 decimal points, 999,999 = 99.9999%)
1724
uint32 public defaultReserveRatio;
1825

26+
// Master copy address that holds implementation of curation token
27+
// This is used as the target for GraphCurationToken clones
28+
address public curationTokenMaster;
29+
1930
// Minimum amount allowed to be deposited by curators to initialize a pool
2031
// This is the `startPoolBalance` for the bonding curve
2132
uint256 public minimumCurationDeposit;
2233

23-
// Bonding curve formula
34+
// Bonding curve library
2435
address public bondingCurve;
2536

2637
// Mapping of subgraphDeploymentID => CurationPool
2738
// There is only one CurationPool per SubgraphDeploymentID
28-
mapping(bytes32 => ICuration.CurationPool) public pools;
39+
mapping(bytes32 => CurationPool) public pools;
2940
}

‎contracts/curation/GraphCurationToken.sol

+9-4
Original file line numberDiff line numberDiff line change
@@ -2,25 +2,30 @@
22

33
pragma solidity ^0.7.6;
44

5-
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
5+
import "@openzeppelin/contracts-upgradeable/token/ERC20/ERC20Upgradeable.sol";
66

77
import "../governance/Governed.sol";
88

99
/**
1010
* @title GraphCurationToken contract
1111
* @dev This is the implementation of the Curation ERC20 token (GCS).
12+
*
1213
* GCS are created for each subgraph deployment curated in the Curation contract.
1314
* The Curation contract is the owner of GCS tokens and the only one allowed to mint or
1415
* burn them. GCS tokens are transferrable and their holders can do any action allowed
1516
* in a standard ERC20 token implementation except for burning them.
17+
*
18+
* This contract is meant to be used as the implementation for Minimal Proxy clones for
19+
* gas-saving purposes.
1620
*/
17-
contract GraphCurationToken is ERC20, Governed {
21+
contract GraphCurationToken is ERC20Upgradeable, Governed {
1822
/**
19-
* @dev Graph Curation Token Contract Constructor.
23+
* @dev Graph Curation Token Contract initializer.
2024
* @param _owner Address of the contract issuing this token
2125
*/
22-
constructor(address _owner) ERC20("Graph Curation Share", "GCS") {
26+
function initialize(address _owner) external initializer {
2327
Governed._initialize(_owner);
28+
ERC20Upgradeable.__ERC20_init("Graph Curation Share", "GCS");
2429
}
2530

2631
/**

‎contracts/curation/ICuration.sol

+2-8
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,6 @@ pragma solidity ^0.7.6;
55
import "./IGraphCurationToken.sol";
66

77
interface ICuration {
8-
// -- Pool --
9-
10-
struct CurationPool {
11-
uint256 tokens; // GRT Tokens stored as reserves for the subgraph deployment
12-
uint32 reserveRatio; // Ratio for the bonding curve
13-
IGraphCurationToken gcs; // Curation token contract for this curation pool
14-
}
15-
168
// -- Configuration --
179

1810
function setDefaultReserveRatio(uint32 _defaultReserveRatio) external;
@@ -21,6 +13,8 @@ interface ICuration {
2113

2214
function setCurationTaxPercentage(uint32 _percentage) external;
2315

16+
function setCurationTokenMaster(address _curationTokenMaster) external;
17+
2418
// -- Curation --
2519

2620
function mint(

‎contracts/curation/IGraphCurationToken.sol

+4-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,11 @@
22

33
pragma solidity ^0.7.6;
44

5-
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
5+
import "@openzeppelin/contracts-upgradeable/token/ERC20/IERC20Upgradeable.sol";
6+
7+
interface IGraphCurationToken is IERC20Upgradeable {
8+
function initialize(address _owner) external;
69

7-
interface IGraphCurationToken is IERC20 {
810
function burnFrom(address _account, uint256 _amount) external;
911

1012
function mint(address _to, uint256 _amount) external;

‎graph.config.yml

+2-1
Original file line numberDiff line numberDiff line change
@@ -39,12 +39,13 @@ contracts:
3939
initialSupply: "10000000000000000000000000000" # 10,000,000,000 GRT
4040
calls:
4141
- fn: "addMinter"
42-
minter: "${{RewardsManager.address}}"
42+
minter: "${{RewardsManager.address}}"
4343
Curation:
4444
proxy: true
4545
init:
4646
controller: "${{Controller.address}}"
4747
bondingCurve: "${{BancorFormula.address}}"
48+
curationTokenMaster: "${{GraphCurationToken.address}}"
4849
reserveRatio: 500000 # 50% (parts per million)
4950
curationTaxPercentage: 25000 # 2.5% (parts per million)
5051
minimumCurationDeposit: "1000000000000000000" # 1 GRT

‎package.json

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
"@nomiclabs/hardhat-etherscan": "^2.1.1",
2626
"@nomiclabs/hardhat-waffle": "^2.0.1",
2727
"@openzeppelin/contracts": "^3.4.1",
28+
"@openzeppelin/contracts-upgradeable": "3.4.2",
2829
"@openzeppelin/hardhat-upgrades": "^1.6.0",
2930
"@tenderly/hardhat-tenderly": "^1.0.11",
3031
"@typechain/ethers-v5": "^7.0.0",

‎test/curation/configuration.test.ts

+29-1
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,13 @@
11
import { expect } from 'chai'
2+
import { constants } from 'ethers'
23

34
import { Curation } from '../../build/types/Curation'
45

56
import { defaults } from '../lib/deployment'
67
import { NetworkFixture } from '../lib/fixtures'
7-
import { getAccounts, toBN, Account } from '../lib/testHelpers'
8+
import { getAccounts, toBN, Account, randomAddress } from '../lib/testHelpers'
9+
10+
const { AddressZero } = constants
811

912
const MAX_PPM = 1000000
1013

@@ -99,4 +102,29 @@ describe('Curation:Config', () => {
99102
await expect(tx).revertedWith('Caller must be Controller governor')
100103
})
101104
})
105+
106+
describe('curationTokenMaster', function () {
107+
it('should set `curationTokenMaster`', async function () {
108+
const newCurationTokenMaster = curation.address
109+
await curation.connect(governor.signer).setCurationTokenMaster(newCurationTokenMaster)
110+
})
111+
112+
it('reject set `curationTokenMaster` to empty value', async function () {
113+
const newCurationTokenMaster = AddressZero
114+
const tx = curation.connect(governor.signer).setCurationTokenMaster(newCurationTokenMaster)
115+
await expect(tx).revertedWith('Token master must be non-empty')
116+
})
117+
118+
it('reject set `curationTokenMaster` to non-contract', async function () {
119+
const newCurationTokenMaster = randomAddress()
120+
const tx = curation.connect(governor.signer).setCurationTokenMaster(newCurationTokenMaster)
121+
await expect(tx).revertedWith('Token master must be a contract')
122+
})
123+
124+
it('reject set `curationTokenMaster` if not allowed', async function () {
125+
const newCurationTokenMaster = curation.address
126+
const tx = curation.connect(me.signer).setCurationTokenMaster(newCurationTokenMaster)
127+
await expect(tx).revertedWith('Caller must be Controller governor')
128+
})
129+
})
102130
})

‎test/curation/curation.test.ts

+16
Original file line numberDiff line numberDiff line change
@@ -456,6 +456,22 @@ describe('Curation', () => {
456456
.burn(subgraphDeploymentID, signalToRedeem, expectedTokens.add(1))
457457
await expect(tx).revertedWith('Slippage protection')
458458
})
459+
460+
it('should not re-deploy the curation token when signal is reset', async function () {
461+
const beforeSubgraphPool = await curation.pools(subgraphDeploymentID)
462+
463+
// Burn all the signal
464+
const signalToRedeem = await curation.getCuratorSignal(curator.address, subgraphDeploymentID)
465+
const expectedTokens = tokensToDeposit
466+
await shouldBurn(signalToRedeem, expectedTokens)
467+
468+
// Mint again on the same subgraph
469+
await curation.connect(curator.signer).mint(subgraphDeploymentID, tokensToDeposit, 0)
470+
471+
// Check state
472+
const afterSubgraphPool = await curation.pools(subgraphDeploymentID)
473+
expect(afterSubgraphPool.gcs).eq(beforeSubgraphPool.gcs)
474+
})
459475
})
460476

461477
describe('conservation', async function () {

‎test/lib/deployment.ts

+2
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,7 @@ export async function deployCuration(
117117
): Promise<Curation> {
118118
// Dependency
119119
const bondingCurve = (await deployContract('BancorFormula', deployer)) as unknown as BancorFormula
120+
const curationTokenMaster = await deployContract('GraphCurationToken', deployer)
120121

121122
// Deploy
122123
return network.deployContractWithProxy(
@@ -125,6 +126,7 @@ export async function deployCuration(
125126
[
126127
controller,
127128
bondingCurve.address,
129+
curationTokenMaster.address,
128130
defaults.curation.reserveRatio,
129131
defaults.curation.curationTaxPercentage,
130132
defaults.curation.minimumCurationDeposit,

‎test/staking/delegation.test.ts

-1
Original file line numberDiff line numberDiff line change
@@ -467,7 +467,6 @@ describe('Staking::Delegation', () => {
467467
await staking.setDelegationUnbondingPeriod('2')
468468
await shouldDelegate(delegator, toGRT('100'))
469469
await shouldUndelegate(delegator, toGRT('50'))
470-
await advanceBlock()
471470
await advanceToNextEpoch(epochManager) // epoch 1
472471
await advanceToNextEpoch(epochManager) // epoch 2
473472
await shouldUndelegate(delegator, toGRT('10'))

‎yarn.lock

+5
Original file line numberDiff line numberDiff line change
@@ -905,6 +905,11 @@
905905
"@types/sinon-chai" "^3.2.3"
906906
"@types/web3" "1.0.19"
907907

908+
"@openzeppelin/contracts-upgradeable@3.4.2":
909+
version "3.4.2"
910+
resolved "https://registry.yarnpkg.com/@openzeppelin/contracts-upgradeable/-/contracts-upgradeable-3.4.2.tgz#2c2a1b0fa748235a1f495b6489349776365c51b3"
911+
integrity sha512-mDlBS17ymb2wpaLcrqRYdnBAmP1EwqhOXMvqWk2c5Q1N1pm5TkiCtXM9Xzznh4bYsQBq0aIWEkFFE2+iLSN1Tw==
912+
908913
"@openzeppelin/contracts@^3.4.1":
909914
version "3.4.2"
910915
resolved "https://registry.yarnpkg.com/@openzeppelin/contracts/-/contracts-3.4.2.tgz#d81f786fda2871d1eb8a8c5a73e455753ba53527"

0 commit comments

Comments
 (0)
Please sign in to comment.