@rev-net/core-v6 0.0.37 → 0.0.39
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +2 -2
- package/README.md +6 -7
- package/foundry.toml +1 -1
- package/package.json +23 -16
- package/references/operations.md +1 -1
- package/references/runtime.md +1 -1
- package/script/Deploy.s.sol +12 -9
- package/src/REVDeployer.sol +60 -65
- package/src/REVHiddenTokens.sol +2 -2
- package/src/REVLoans.sol +17 -10
- package/src/REVOwner.sol +121 -14
- package/src/interfaces/IREVDeployer.sol +2 -1
- package/src/interfaces/IREVHiddenTokens.sol +4 -1
- package/src/interfaces/IREVOwner.sol +5 -0
- package/ADMINISTRATION.md +0 -73
- package/ARCHITECTURE.md +0 -116
- package/AUDIT_INSTRUCTIONS.md +0 -90
- package/RISKS.md +0 -107
- package/SKILLS.md +0 -46
- package/STYLE_GUIDE.md +0 -610
- package/USER_JOURNEYS.md +0 -195
- package/foundry.lock +0 -11
- package/slither-ci.config.json +0 -10
- package/sphinx.lock +0 -507
- package/test/REV.integrations.t.sol +0 -573
- package/test/REVAutoIssuanceFuzz.t.sol +0 -328
- package/test/REVDeployerRegressions.t.sol +0 -396
- package/test/REVInvincibility.t.sol +0 -1371
- package/test/REVInvincibilityHandler.sol +0 -387
- package/test/REVLifecycle.t.sol +0 -420
- package/test/REVLoans.invariants.t.sol +0 -724
- package/test/REVLoansAttacks.t.sol +0 -816
- package/test/REVLoansFeeRecovery.t.sol +0 -783
- package/test/REVLoansFindings.t.sol +0 -711
- package/test/REVLoansRegressions.t.sol +0 -364
- package/test/REVLoansSourceFeeRecovery.t.sol +0 -517
- package/test/REVLoansSourced.t.sol +0 -1839
- package/test/REVLoansUnSourced.t.sol +0 -409
- package/test/TestAuditFixVerification.t.sol +0 -675
- package/test/TestBurnHeldTokens.t.sol +0 -394
- package/test/TestCEIPattern.t.sol +0 -508
- package/test/TestCashOutCallerValidation.t.sol +0 -452
- package/test/TestConversionDocumentation.t.sol +0 -365
- package/test/TestCrossCurrencyReclaim.t.sol +0 -610
- package/test/TestCrossSourceReallocation.t.sol +0 -361
- package/test/TestERC2771MetaTx.t.sol +0 -585
- package/test/TestEmptyBuybackSpecs.t.sol +0 -300
- package/test/TestFlashLoanSurplus.t.sol +0 -365
- package/test/TestHiddenTokens.t.sol +0 -474
- package/test/TestHookArrayOOB.t.sol +0 -278
- package/test/TestLiquidationBehavior.t.sol +0 -398
- package/test/TestLoanSourceRotation.t.sol +0 -553
- package/test/TestLoansCashOutDelay.t.sol +0 -493
- package/test/TestLongTailEconomics.t.sol +0 -677
- package/test/TestLowFindings.t.sol +0 -677
- package/test/TestMixedFixes.t.sol +0 -593
- package/test/TestPermit2Signatures.t.sol +0 -683
- package/test/TestReallocationSandwich.t.sol +0 -412
- package/test/TestRevnetRegressions.t.sol +0 -350
- package/test/TestSplitWeightAdjustment.t.sol +0 -527
- package/test/TestSplitWeightE2E.t.sol +0 -605
- package/test/TestSplitWeightFork.t.sol +0 -855
- package/test/TestStageTransitionBorrowable.t.sol +0 -301
- package/test/TestSwapTerminalPermission.t.sol +0 -262
- package/test/TestTerminalEncodingInHash.t.sol +0 -326
- package/test/TestUint112Overflow.t.sol +0 -311
- package/test/TestZeroAmountLoanGuard.t.sol +0 -378
- package/test/TestZeroRepayment.t.sol +0 -354
- package/test/audit/CrossChainBuybackRouteMismatch.t.sol +0 -184
- package/test/audit/HiddenSupplyCashout.t.sol +0 -61
- package/test/audit/LoanIdOverflowGuard.t.sol +0 -523
- package/test/audit/NemesisVerification.t.sol +0 -97
- package/test/audit/OperatorDelegation.t.sol +0 -356
- package/test/audit/PhantomSurplusTerminal.t.sol +0 -367
- package/test/audit/REVOwnerCurrencyMismatch.t.sol +0 -188
- package/test/audit/REVOwnerRemoteSurplusCurrencyMismatch.t.sol +0 -140
- package/test/audit/ReallocatePermission.t.sol +0 -363
- package/test/audit/RemoteLoanAccountingGap.t.sol +0 -74
- package/test/audit/SupportsInterfaceTest.t.sol +0 -51
- package/test/audit/TestFeeAllowanceLeak.t.sol +0 -197
- package/test/audit/TestLoansAndDeployerFixes.t.sol +0 -576
- package/test/fork/ForkTestBase.sol +0 -727
- package/test/fork/TestAutoIssuanceFork.t.sol +0 -148
- package/test/fork/TestCashOutFork.t.sol +0 -253
- package/test/fork/TestIssuanceDecayFork.t.sol +0 -158
- package/test/fork/TestLoanAdversarialFork.t.sol +0 -744
- package/test/fork/TestLoanBorrowFork.t.sol +0 -163
- package/test/fork/TestLoanCrossRulesetFork.t.sol +0 -308
- package/test/fork/TestLoanERC20Fork.t.sol +0 -459
- package/test/fork/TestLoanLiquidationFork.t.sol +0 -135
- package/test/fork/TestLoanReallocateFork.t.sol +0 -113
- package/test/fork/TestLoanRepayFork.t.sol +0 -188
- package/test/fork/TestLoanTransferFork.t.sol +0 -143
- package/test/fork/TestPermit2PaymentFork.t.sol +0 -300
- package/test/fork/TestSplitWeightFork.t.sol +0 -189
- package/test/helpers/MaliciousContracts.sol +0 -247
- package/test/helpers/REVEmpty721Config.sol +0 -45
- package/test/mock/MockBuybackCashOutRecorder.sol +0 -84
- package/test/mock/MockBuybackDataHook.sol +0 -112
- package/test/mock/MockBuybackDataHookMintPath.sol +0 -68
- package/test/mock/MockSuckerRegistry.sol +0 -17
- package/test/regression/TestBurnPermissionRequired.t.sol +0 -294
- package/test/regression/TestCashOutBuybackFeeLeak.t.sol +0 -232
- package/test/regression/TestCrossRevnetLiquidation.t.sol +0 -255
- package/test/regression/TestCumulativeLoanCounter.t.sol +0 -361
- package/test/regression/TestLiquidateGapHandling.t.sol +0 -394
- package/test/regression/TestZeroPriceFeed.t.sol +0 -422
|
@@ -1,474 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: MIT
|
|
2
|
-
pragma solidity 0.8.28;
|
|
3
|
-
|
|
4
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
5
|
-
import "forge-std/Test.sol";
|
|
6
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
7
|
-
import /* {*} from */ "@bananapus/core-v6/test/helpers/TestBaseWorkflow.sol";
|
|
8
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
9
|
-
import /* {*} from */ "./../src/REVDeployer.sol";
|
|
10
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
11
|
-
import "@croptop/core-v6/src/CTPublisher.sol";
|
|
12
|
-
import {MockBuybackDataHook} from "./mock/MockBuybackDataHook.sol";
|
|
13
|
-
import {REVEmpty721Config} from "./helpers/REVEmpty721Config.sol";
|
|
14
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
15
|
-
import "@bananapus/core-v6/script/helpers/CoreDeploymentLib.sol";
|
|
16
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
17
|
-
import "@bananapus/721-hook-v6/script/helpers/Hook721DeploymentLib.sol";
|
|
18
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
19
|
-
import "@bananapus/suckers-v6/script/helpers/SuckerDeploymentLib.sol";
|
|
20
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
21
|
-
import "@croptop/core-v6/script/helpers/CroptopDeploymentLib.sol";
|
|
22
|
-
// forge-lint: disable-next-line(unaliased-plain-import)
|
|
23
|
-
import "@bananapus/router-terminal-v6/script/helpers/RouterTerminalDeploymentLib.sol";
|
|
24
|
-
import {JBConstants} from "@bananapus/core-v6/src/libraries/JBConstants.sol";
|
|
25
|
-
import {JBPermissionIds} from "@bananapus/permission-ids-v6/src/JBPermissionIds.sol";
|
|
26
|
-
import {JBAccountingContext} from "@bananapus/core-v6/src/structs/JBAccountingContext.sol";
|
|
27
|
-
import {JBPermissionsData} from "@bananapus/core-v6/src/structs/JBPermissionsData.sol";
|
|
28
|
-
import {MockERC20} from "@bananapus/core-v6/test/mock/MockERC20.sol";
|
|
29
|
-
import {REVLoans} from "../src/REVLoans.sol";
|
|
30
|
-
import {REVHiddenTokens} from "../src/REVHiddenTokens.sol";
|
|
31
|
-
import {IREVHiddenTokens} from "../src/interfaces/IREVHiddenTokens.sol";
|
|
32
|
-
import {REVStageConfig, REVAutoIssuance} from "../src/structs/REVStageConfig.sol";
|
|
33
|
-
import {REVDescription} from "../src/structs/REVDescription.sol";
|
|
34
|
-
import {IREVLoans} from "./../src/interfaces/IREVLoans.sol";
|
|
35
|
-
import {JBSuckerDeployerConfig} from "@bananapus/suckers-v6/src/structs/JBSuckerDeployerConfig.sol";
|
|
36
|
-
import {JBSuckerRegistry} from "@bananapus/suckers-v6/src/JBSuckerRegistry.sol";
|
|
37
|
-
import {JB721TiersHookDeployer} from "@bananapus/721-hook-v6/src/JB721TiersHookDeployer.sol";
|
|
38
|
-
import {JB721TiersHook} from "@bananapus/721-hook-v6/src/JB721TiersHook.sol";
|
|
39
|
-
import {JB721TiersHookStore} from "@bananapus/721-hook-v6/src/JB721TiersHookStore.sol";
|
|
40
|
-
import {JB721CheckpointsDeployer} from "@bananapus/721-hook-v6/src/JB721CheckpointsDeployer.sol";
|
|
41
|
-
import {IJB721CheckpointsDeployer} from "@bananapus/721-hook-v6/src/interfaces/IJB721CheckpointsDeployer.sol";
|
|
42
|
-
import {JBAddressRegistry} from "@bananapus/address-registry-v6/src/JBAddressRegistry.sol";
|
|
43
|
-
import {IJBAddressRegistry} from "@bananapus/address-registry-v6/src/interfaces/IJBAddressRegistry.sol";
|
|
44
|
-
import {REVOwner} from "../src/REVOwner.sol";
|
|
45
|
-
import {IREVDeployer} from "../src/interfaces/IREVDeployer.sol";
|
|
46
|
-
import {MockSuckerRegistry} from "./mock/MockSuckerRegistry.sol";
|
|
47
|
-
|
|
48
|
-
/// @notice Tests for the standalone REVHiddenTokens contract.
|
|
49
|
-
contract TestHiddenTokens is TestBaseWorkflow {
|
|
50
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
51
|
-
bytes32 REV_DEPLOYER_SALT = "REVDeployer";
|
|
52
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
53
|
-
bytes32 ERC20_SALT = "REV_TOKEN";
|
|
54
|
-
|
|
55
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
56
|
-
REVDeployer REV_DEPLOYER;
|
|
57
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
58
|
-
REVOwner REV_OWNER;
|
|
59
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
60
|
-
JB721TiersHook EXAMPLE_HOOK;
|
|
61
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
62
|
-
IJB721TiersHookDeployer HOOK_DEPLOYER;
|
|
63
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
64
|
-
IJB721TiersHookStore HOOK_STORE;
|
|
65
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
66
|
-
IJBAddressRegistry ADDRESS_REGISTRY;
|
|
67
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
68
|
-
IREVLoans LOANS_CONTRACT;
|
|
69
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
70
|
-
REVHiddenTokens HIDDEN_TOKENS;
|
|
71
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
72
|
-
IJBSuckerRegistry SUCKER_REGISTRY;
|
|
73
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
74
|
-
CTPublisher PUBLISHER;
|
|
75
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
76
|
-
MockBuybackDataHook MOCK_BUYBACK;
|
|
77
|
-
|
|
78
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
79
|
-
uint256 FEE_PROJECT_ID;
|
|
80
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
81
|
-
uint256 REVNET_ID;
|
|
82
|
-
|
|
83
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
84
|
-
address USER = makeAddr("user");
|
|
85
|
-
// forge-lint: disable-next-line(mixed-case-variable)
|
|
86
|
-
address BENEFICIARY = makeAddr("beneficiary");
|
|
87
|
-
|
|
88
|
-
address private constant TRUSTED_FORWARDER = 0xB2b5841DBeF766d4b521221732F9B618fCf34A87;
|
|
89
|
-
|
|
90
|
-
function setUp() public override {
|
|
91
|
-
super.setUp();
|
|
92
|
-
FEE_PROJECT_ID = jbProjects().createFor(multisig());
|
|
93
|
-
SUCKER_REGISTRY = new JBSuckerRegistry(jbDirectory(), jbPermissions(), multisig(), address(0));
|
|
94
|
-
HOOK_STORE = new JB721TiersHookStore();
|
|
95
|
-
EXAMPLE_HOOK = new JB721TiersHook(
|
|
96
|
-
jbDirectory(),
|
|
97
|
-
jbPermissions(),
|
|
98
|
-
jbPrices(),
|
|
99
|
-
jbRulesets(),
|
|
100
|
-
HOOK_STORE,
|
|
101
|
-
jbSplits(),
|
|
102
|
-
IJB721CheckpointsDeployer(address(new JB721CheckpointsDeployer())),
|
|
103
|
-
multisig()
|
|
104
|
-
);
|
|
105
|
-
ADDRESS_REGISTRY = new JBAddressRegistry();
|
|
106
|
-
HOOK_DEPLOYER = new JB721TiersHookDeployer(EXAMPLE_HOOK, HOOK_STORE, ADDRESS_REGISTRY, multisig());
|
|
107
|
-
PUBLISHER = new CTPublisher(jbDirectory(), jbPermissions(), FEE_PROJECT_ID, multisig());
|
|
108
|
-
MOCK_BUYBACK = new MockBuybackDataHook();
|
|
109
|
-
|
|
110
|
-
LOANS_CONTRACT = new REVLoans({
|
|
111
|
-
controller: jbController(),
|
|
112
|
-
suckerRegistry: IJBSuckerRegistry(address(new MockSuckerRegistry())),
|
|
113
|
-
revId: FEE_PROJECT_ID,
|
|
114
|
-
owner: address(this),
|
|
115
|
-
permit2: permit2(),
|
|
116
|
-
trustedForwarder: TRUSTED_FORWARDER
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
HIDDEN_TOKENS = new REVHiddenTokens(jbController(), TRUSTED_FORWARDER);
|
|
120
|
-
|
|
121
|
-
REV_OWNER = new REVOwner(
|
|
122
|
-
IJBBuybackHookRegistry(address(MOCK_BUYBACK)),
|
|
123
|
-
jbDirectory(),
|
|
124
|
-
FEE_PROJECT_ID,
|
|
125
|
-
SUCKER_REGISTRY,
|
|
126
|
-
address(LOANS_CONTRACT),
|
|
127
|
-
address(HIDDEN_TOKENS)
|
|
128
|
-
);
|
|
129
|
-
|
|
130
|
-
REV_DEPLOYER = new REVDeployer{salt: REV_DEPLOYER_SALT}(
|
|
131
|
-
jbController(),
|
|
132
|
-
SUCKER_REGISTRY,
|
|
133
|
-
FEE_PROJECT_ID,
|
|
134
|
-
HOOK_DEPLOYER,
|
|
135
|
-
PUBLISHER,
|
|
136
|
-
IJBBuybackHookRegistry(address(MOCK_BUYBACK)),
|
|
137
|
-
address(LOANS_CONTRACT),
|
|
138
|
-
TRUSTED_FORWARDER,
|
|
139
|
-
address(REV_OWNER)
|
|
140
|
-
);
|
|
141
|
-
|
|
142
|
-
REV_OWNER.setDeployer(REV_DEPLOYER);
|
|
143
|
-
|
|
144
|
-
vm.prank(multisig());
|
|
145
|
-
jbProjects().approve(address(REV_DEPLOYER), FEE_PROJECT_ID);
|
|
146
|
-
_deployFeeProject();
|
|
147
|
-
REVNET_ID = _deployRevnet();
|
|
148
|
-
vm.deal(USER, 100e18);
|
|
149
|
-
_grantBurnPermission(USER, REVNET_ID);
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
// ──────────────────── Test: Hiding reduces totalSupply
|
|
153
|
-
// ────────────────────
|
|
154
|
-
|
|
155
|
-
function test_hideTokens_reducesTotalSupply() public {
|
|
156
|
-
// Pay to get tokens.
|
|
157
|
-
uint256 payAmount = 10e18;
|
|
158
|
-
vm.prank(USER);
|
|
159
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
160
|
-
projectId: REVNET_ID,
|
|
161
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
162
|
-
amount: payAmount,
|
|
163
|
-
beneficiary: USER,
|
|
164
|
-
minReturnedTokens: 0,
|
|
165
|
-
memo: "",
|
|
166
|
-
metadata: ""
|
|
167
|
-
});
|
|
168
|
-
|
|
169
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
170
|
-
assertGt(userTokens, 0, "User should have tokens after paying");
|
|
171
|
-
|
|
172
|
-
uint256 totalSupplyBefore = jbController().TOKENS().totalSupplyOf(REVNET_ID);
|
|
173
|
-
|
|
174
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
175
|
-
|
|
176
|
-
// Hide half the tokens.
|
|
177
|
-
uint256 hideCount = userTokens / 2;
|
|
178
|
-
vm.prank(USER);
|
|
179
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, hideCount, USER);
|
|
180
|
-
|
|
181
|
-
uint256 totalSupplyAfter = jbController().TOKENS().totalSupplyOf(REVNET_ID);
|
|
182
|
-
assertEq(totalSupplyAfter, totalSupplyBefore - hideCount, "Total supply should decrease by hidden amount");
|
|
183
|
-
assertEq(HIDDEN_TOKENS.hiddenBalanceOf(USER, REVNET_ID), hideCount, "Hidden balance should match");
|
|
184
|
-
assertEq(HIDDEN_TOKENS.totalHiddenOf(REVNET_ID), hideCount, "Total hidden should match");
|
|
185
|
-
}
|
|
186
|
-
|
|
187
|
-
// ──────────────────── Test: Revealing restores tokens
|
|
188
|
-
// ────────────────────
|
|
189
|
-
|
|
190
|
-
function test_revealTokens_restoresTokens() public {
|
|
191
|
-
// Pay to get tokens.
|
|
192
|
-
uint256 payAmount = 10e18;
|
|
193
|
-
vm.prank(USER);
|
|
194
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
195
|
-
projectId: REVNET_ID,
|
|
196
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
197
|
-
amount: payAmount,
|
|
198
|
-
beneficiary: USER,
|
|
199
|
-
minReturnedTokens: 0,
|
|
200
|
-
memo: "",
|
|
201
|
-
metadata: ""
|
|
202
|
-
});
|
|
203
|
-
|
|
204
|
-
uint256 userTokensBefore = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
205
|
-
uint256 totalSupplyBefore = jbController().TOKENS().totalSupplyOf(REVNET_ID);
|
|
206
|
-
|
|
207
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
208
|
-
|
|
209
|
-
// Hide tokens.
|
|
210
|
-
uint256 hideCount = userTokensBefore / 2;
|
|
211
|
-
vm.prank(USER);
|
|
212
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, hideCount, USER);
|
|
213
|
-
|
|
214
|
-
// Reveal tokens back to USER.
|
|
215
|
-
vm.prank(USER);
|
|
216
|
-
HIDDEN_TOKENS.revealTokensOf(REVNET_ID, hideCount, USER);
|
|
217
|
-
|
|
218
|
-
uint256 totalSupplyAfter = jbController().TOKENS().totalSupplyOf(REVNET_ID);
|
|
219
|
-
assertEq(totalSupplyAfter, totalSupplyBefore, "Total supply should be restored");
|
|
220
|
-
assertEq(HIDDEN_TOKENS.hiddenBalanceOf(USER, REVNET_ID), 0, "Hidden balance should be zero");
|
|
221
|
-
assertEq(HIDDEN_TOKENS.totalHiddenOf(REVNET_ID), 0, "Total hidden should be zero");
|
|
222
|
-
assertEq(
|
|
223
|
-
jbController().TOKENS().totalBalanceOf(USER, REVNET_ID), userTokensBefore, "User should receive tokens"
|
|
224
|
-
);
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
// ──────────────────── Test: Insufficient hidden balance reverts
|
|
228
|
-
// ────────────────────
|
|
229
|
-
|
|
230
|
-
function test_revealTokens_revertsOnInsufficientBalance() public {
|
|
231
|
-
// Pay to get tokens.
|
|
232
|
-
uint256 payAmount = 10e18;
|
|
233
|
-
vm.prank(USER);
|
|
234
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
235
|
-
projectId: REVNET_ID,
|
|
236
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
237
|
-
amount: payAmount,
|
|
238
|
-
beneficiary: USER,
|
|
239
|
-
minReturnedTokens: 0,
|
|
240
|
-
memo: "",
|
|
241
|
-
metadata: ""
|
|
242
|
-
});
|
|
243
|
-
|
|
244
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
245
|
-
uint256 hideCount = userTokens / 4;
|
|
246
|
-
|
|
247
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
248
|
-
|
|
249
|
-
// Hide some tokens.
|
|
250
|
-
vm.prank(USER);
|
|
251
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, hideCount, USER);
|
|
252
|
-
|
|
253
|
-
// Try to reveal more than hidden — should revert.
|
|
254
|
-
vm.prank(USER);
|
|
255
|
-
vm.expectRevert(
|
|
256
|
-
abi.encodeWithSelector(
|
|
257
|
-
REVHiddenTokens.REVHiddenTokens_InsufficientHiddenBalance.selector, hideCount, hideCount + 1
|
|
258
|
-
)
|
|
259
|
-
);
|
|
260
|
-
HIDDEN_TOKENS.revealTokensOf(REVNET_ID, hideCount + 1, USER);
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
// ──────────────────── Test: Hidden tokens inflate cash out rate
|
|
264
|
-
// ────────────────────
|
|
265
|
-
|
|
266
|
-
function test_hiddenTokens_inflateCashOutRate() public {
|
|
267
|
-
// Pay to get tokens for 2 users.
|
|
268
|
-
uint256 payAmount = 10e18;
|
|
269
|
-
vm.prank(USER);
|
|
270
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
271
|
-
projectId: REVNET_ID,
|
|
272
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
273
|
-
amount: payAmount,
|
|
274
|
-
beneficiary: USER,
|
|
275
|
-
minReturnedTokens: 0,
|
|
276
|
-
memo: "",
|
|
277
|
-
metadata: ""
|
|
278
|
-
});
|
|
279
|
-
|
|
280
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
281
|
-
|
|
282
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
283
|
-
|
|
284
|
-
// Hide half the user's tokens.
|
|
285
|
-
uint256 hideCount = userTokens / 2;
|
|
286
|
-
vm.prank(USER);
|
|
287
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, hideCount, USER);
|
|
288
|
-
|
|
289
|
-
// The remaining tokens now represent a larger share of totalSupply.
|
|
290
|
-
uint256 totalSupply = jbController().TOKENS().totalSupplyOf(REVNET_ID);
|
|
291
|
-
uint256 remainingBalance = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
292
|
-
assertEq(remainingBalance, userTokens - hideCount, "Remaining balance should be half");
|
|
293
|
-
assertEq(totalSupply, userTokens - hideCount, "Total supply should equal remaining balance");
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
// ──────────────────── Test: Events emitted correctly
|
|
297
|
-
// ────────────────────
|
|
298
|
-
|
|
299
|
-
function test_hideTokens_emitsEvent() public {
|
|
300
|
-
uint256 payAmount = 10e18;
|
|
301
|
-
vm.prank(USER);
|
|
302
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
303
|
-
projectId: REVNET_ID,
|
|
304
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
305
|
-
amount: payAmount,
|
|
306
|
-
beneficiary: USER,
|
|
307
|
-
minReturnedTokens: 0,
|
|
308
|
-
memo: "",
|
|
309
|
-
metadata: ""
|
|
310
|
-
});
|
|
311
|
-
|
|
312
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
313
|
-
|
|
314
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
315
|
-
|
|
316
|
-
vm.prank(USER);
|
|
317
|
-
vm.expectEmit(true, false, false, true);
|
|
318
|
-
emit IREVHiddenTokens.HideTokens(REVNET_ID, userTokens, USER, USER);
|
|
319
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, userTokens, USER);
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
function test_revealTokens_emitsEvent() public {
|
|
323
|
-
uint256 payAmount = 10e18;
|
|
324
|
-
vm.prank(USER);
|
|
325
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
326
|
-
projectId: REVNET_ID,
|
|
327
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
328
|
-
amount: payAmount,
|
|
329
|
-
beneficiary: USER,
|
|
330
|
-
minReturnedTokens: 0,
|
|
331
|
-
memo: "",
|
|
332
|
-
metadata: ""
|
|
333
|
-
});
|
|
334
|
-
|
|
335
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
336
|
-
|
|
337
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
338
|
-
|
|
339
|
-
vm.prank(USER);
|
|
340
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, userTokens, USER);
|
|
341
|
-
|
|
342
|
-
vm.prank(USER);
|
|
343
|
-
vm.expectEmit(true, false, false, true);
|
|
344
|
-
emit IREVHiddenTokens.RevealTokens(REVNET_ID, userTokens, USER, USER);
|
|
345
|
-
HIDDEN_TOKENS.revealTokensOf(REVNET_ID, userTokens, USER);
|
|
346
|
-
}
|
|
347
|
-
|
|
348
|
-
function test_setTokenHidingAllowedFor_allowsHolderToHide() public {
|
|
349
|
-
uint256 payAmount = 10e18;
|
|
350
|
-
|
|
351
|
-
vm.prank(USER);
|
|
352
|
-
jbMultiTerminal().pay{value: payAmount}({
|
|
353
|
-
projectId: REVNET_ID,
|
|
354
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
355
|
-
amount: payAmount,
|
|
356
|
-
beneficiary: USER,
|
|
357
|
-
minReturnedTokens: 0,
|
|
358
|
-
memo: "",
|
|
359
|
-
metadata: ""
|
|
360
|
-
});
|
|
361
|
-
|
|
362
|
-
uint256 userTokens = jbController().TOKENS().totalBalanceOf(USER, REVNET_ID);
|
|
363
|
-
uint256 hideCount = userTokens / 2;
|
|
364
|
-
|
|
365
|
-
_allowHolderToHide(USER, REVNET_ID);
|
|
366
|
-
|
|
367
|
-
vm.prank(USER);
|
|
368
|
-
HIDDEN_TOKENS.hideTokensOf(REVNET_ID, hideCount, USER);
|
|
369
|
-
|
|
370
|
-
vm.prank(USER);
|
|
371
|
-
HIDDEN_TOKENS.revealTokensOf(REVNET_ID, hideCount, USER);
|
|
372
|
-
|
|
373
|
-
assertEq(jbController().TOKENS().totalBalanceOf(USER, REVNET_ID), userTokens, "User balance should be restored");
|
|
374
|
-
}
|
|
375
|
-
|
|
376
|
-
// ──────────────────── Internal helpers
|
|
377
|
-
// ────────────────────
|
|
378
|
-
|
|
379
|
-
function _grantBurnPermission(address account, uint256 revnetId) internal {
|
|
380
|
-
uint8[] memory permissionIds = new uint8[](1);
|
|
381
|
-
permissionIds[0] = JBPermissionIds.BURN_TOKENS;
|
|
382
|
-
JBPermissionsData memory permissionsData = JBPermissionsData({
|
|
383
|
-
operator: address(HIDDEN_TOKENS),
|
|
384
|
-
// forge-lint: disable-next-line(unsafe-typecast)
|
|
385
|
-
projectId: uint56(revnetId),
|
|
386
|
-
permissionIds: permissionIds
|
|
387
|
-
});
|
|
388
|
-
vm.prank(account);
|
|
389
|
-
jbPermissions().setPermissionsFor(account, permissionsData);
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
function _allowHolderToHide(address holder, uint256 revnetId) internal {
|
|
393
|
-
vm.prank(address(REV_DEPLOYER));
|
|
394
|
-
HIDDEN_TOKENS.setTokenHidingAllowedFor(revnetId, holder, true);
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
function _deployFeeProject() internal {
|
|
398
|
-
JBAccountingContext[] memory acc = new JBAccountingContext[](1);
|
|
399
|
-
acc[0] = JBAccountingContext({
|
|
400
|
-
token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: uint32(uint160(JBConstants.NATIVE_TOKEN))
|
|
401
|
-
});
|
|
402
|
-
JBTerminalConfig[] memory tc = new JBTerminalConfig[](1);
|
|
403
|
-
tc[0] = JBTerminalConfig({terminal: jbMultiTerminal(), accountingContextsToAccept: acc});
|
|
404
|
-
REVStageConfig[] memory stages = new REVStageConfig[](1);
|
|
405
|
-
stages[0] = REVStageConfig({
|
|
406
|
-
startsAtOrAfter: uint48(block.timestamp),
|
|
407
|
-
autoIssuances: new REVAutoIssuance[](0),
|
|
408
|
-
splitPercent: 0,
|
|
409
|
-
splits: new JBSplit[](0),
|
|
410
|
-
initialIssuance: uint112(1000e18),
|
|
411
|
-
issuanceCutFrequency: 0,
|
|
412
|
-
issuanceCutPercent: 0,
|
|
413
|
-
cashOutTaxRate: 0,
|
|
414
|
-
extraMetadata: 0
|
|
415
|
-
});
|
|
416
|
-
// forge-lint: disable-next-line(named-struct-fields)
|
|
417
|
-
REVConfig memory feeConfig = REVConfig({
|
|
418
|
-
description: REVDescription("Fee Revnet", "FEE", "", ERC20_SALT),
|
|
419
|
-
baseCurrency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
420
|
-
splitOperator: multisig(),
|
|
421
|
-
stageConfigurations: stages
|
|
422
|
-
});
|
|
423
|
-
vm.prank(multisig());
|
|
424
|
-
REV_DEPLOYER.deployFor({
|
|
425
|
-
revnetId: FEE_PROJECT_ID,
|
|
426
|
-
configuration: feeConfig,
|
|
427
|
-
terminalConfigurations: tc,
|
|
428
|
-
suckerDeploymentConfiguration: REVSuckerDeploymentConfig({
|
|
429
|
-
deployerConfigurations: new JBSuckerDeployerConfig[](0), salt: keccak256("FEE")
|
|
430
|
-
}),
|
|
431
|
-
tiered721HookConfiguration: REVEmpty721Config.empty721Config(uint32(uint160(JBConstants.NATIVE_TOKEN))),
|
|
432
|
-
allowedPosts: REVEmpty721Config.emptyAllowedPosts()
|
|
433
|
-
});
|
|
434
|
-
}
|
|
435
|
-
|
|
436
|
-
function _deployRevnet() internal returns (uint256) {
|
|
437
|
-
JBAccountingContext[] memory acc = new JBAccountingContext[](1);
|
|
438
|
-
acc[0] = JBAccountingContext({
|
|
439
|
-
token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: uint32(uint160(JBConstants.NATIVE_TOKEN))
|
|
440
|
-
});
|
|
441
|
-
JBTerminalConfig[] memory tc = new JBTerminalConfig[](1);
|
|
442
|
-
tc[0] = JBTerminalConfig({terminal: jbMultiTerminal(), accountingContextsToAccept: acc});
|
|
443
|
-
REVStageConfig[] memory stages = new REVStageConfig[](1);
|
|
444
|
-
stages[0] = REVStageConfig({
|
|
445
|
-
startsAtOrAfter: uint48(block.timestamp),
|
|
446
|
-
autoIssuances: new REVAutoIssuance[](0),
|
|
447
|
-
splitPercent: 0,
|
|
448
|
-
splits: new JBSplit[](0),
|
|
449
|
-
initialIssuance: uint112(1000e18),
|
|
450
|
-
issuanceCutFrequency: 0,
|
|
451
|
-
issuanceCutPercent: 0,
|
|
452
|
-
cashOutTaxRate: 5000, // 50% cash out tax rate
|
|
453
|
-
extraMetadata: 0
|
|
454
|
-
});
|
|
455
|
-
// forge-lint: disable-next-line(named-struct-fields)
|
|
456
|
-
REVConfig memory revConfig = REVConfig({
|
|
457
|
-
description: REVDescription("Test Revnet", "TEST", "", bytes32("TEST_TOKEN")),
|
|
458
|
-
baseCurrency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
459
|
-
splitOperator: multisig(),
|
|
460
|
-
stageConfigurations: stages
|
|
461
|
-
});
|
|
462
|
-
(uint256 revnetId,) = REV_DEPLOYER.deployFor({
|
|
463
|
-
revnetId: 0,
|
|
464
|
-
configuration: revConfig,
|
|
465
|
-
terminalConfigurations: tc,
|
|
466
|
-
suckerDeploymentConfiguration: REVSuckerDeploymentConfig({
|
|
467
|
-
deployerConfigurations: new JBSuckerDeployerConfig[](0), salt: keccak256("NANA")
|
|
468
|
-
}),
|
|
469
|
-
tiered721HookConfiguration: REVEmpty721Config.empty721Config(uint32(uint160(JBConstants.NATIVE_TOKEN))),
|
|
470
|
-
allowedPosts: REVEmpty721Config.emptyAllowedPosts()
|
|
471
|
-
});
|
|
472
|
-
return revnetId;
|
|
473
|
-
}
|
|
474
|
-
}
|