@bananapus/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/foundry.lock +1 -7
- package/foundry.toml +1 -1
- package/package.json +19 -7
- package/src/JBChainlinkV3PriceFeed.sol +4 -1
- package/src/JBChainlinkV3SequencerPriceFeed.sol +4 -2
- package/src/JBController.sol +71 -44
- package/src/JBDeadline.sol +4 -4
- package/src/JBDirectory.sol +34 -32
- package/src/JBERC20.sol +5 -4
- package/src/JBFeelessAddresses.sol +6 -3
- package/src/JBFundAccessLimits.sol +25 -21
- package/src/JBMultiTerminal.sol +121 -84
- package/src/JBPermissions.sol +34 -37
- package/src/JBPrices.sol +23 -18
- package/src/JBProjects.sol +6 -3
- package/src/JBRulesets.sol +44 -41
- package/src/JBSplits.sol +18 -16
- package/src/JBTerminalStore.sol +32 -25
- package/src/JBTokens.sol +36 -26
- package/src/abstract/JBControlled.sol +3 -1
- package/src/abstract/JBPermissioned.sol +3 -1
- package/src/enums/JBApprovalStatus.sol +7 -1
- package/src/interfaces/IJBController.sol +7 -3
- package/src/interfaces/IJBDirectory.sol +3 -1
- package/src/interfaces/IJBMultiTerminal.sol +3 -2
- package/src/interfaces/IJBPermissions.sol +2 -1
- package/src/interfaces/IJBPrices.sol +3 -1
- package/src/interfaces/IJBRulesets.sol +2 -1
- package/src/interfaces/IJBSplits.sol +2 -1
- package/src/interfaces/IJBTerminal.sol +3 -1
- package/src/interfaces/IJBTerminalStore.sol +3 -1
- package/src/interfaces/IJBTokens.sol +2 -1
- package/src/libraries/JBCashOuts.sol +6 -1
- package/src/libraries/JBConstants.sol +12 -3
- package/src/libraries/JBCurrencyIds.sol +2 -0
- package/src/libraries/JBFees.sol +52 -10
- package/src/libraries/JBFixedPointNumber.sol +2 -0
- package/src/libraries/JBPayoutSplitGroupLib.sol +7 -4
- package/src/libraries/JBRulesetMetadataResolver.sol +4 -0
- package/src/libraries/JBSplitGroupIds.sol +2 -1
- package/src/libraries/JBSurplus.sol +3 -1
- package/src/periphery/JBMatchingPriceFeed.sol +3 -1
- package/src/structs/JBAccountingContext.sol +7 -4
- package/src/structs/JBFundAccessLimitGroup.sol +10 -17
- package/src/structs/JBRuleset.sol +18 -26
- package/src/structs/JBRulesetConfig.sol +13 -25
- package/src/structs/JBRulesetMetadata.sol +25 -32
- package/test/mock/MockMaliciousBeneficiary.sol +15 -15
- package/ADMINISTRATION.md +0 -103
- package/ARCHITECTURE.md +0 -133
- package/AUDIT_INSTRUCTIONS.md +0 -139
- package/RISKS.md +0 -215
- package/SKILLS.md +0 -55
- package/STYLE_GUIDE.md +0 -610
- package/USER_JOURNEYS.md +0 -215
- package/script/Deploy.s.sol +0 -124
- package/script/DeployPeriphery.s.sol +0 -354
- package/slither-ci.config.json +0 -10
- package/test/AuditFixes.t.sol +0 -808
- package/test/ComprehensiveInvariant.t.sol +0 -306
- package/test/CoreExploitTests.t.sol +0 -2741
- package/test/EconomicSimulation.t.sol +0 -348
- package/test/EntryPointPermutations.t.sol +0 -684
- package/test/FlashLoanAttacks.t.sol +0 -797
- package/test/PermissionEscalation.t.sol +0 -711
- package/test/PermissionsInvariant.t.sol +0 -403
- package/test/RulesetTransitions.t.sol +0 -713
- package/test/SplitLoopTests.t.sol +0 -752
- package/test/TestAccessToFunds.sol +0 -2683
- package/test/TestAuditResponseDesignProofs.sol +0 -434
- package/test/TestCashOut.sol +0 -198
- package/test/TestCashOutCountFor.sol +0 -271
- package/test/TestCashOutHooks.sol +0 -351
- package/test/TestCashOutTimingEdge.sol +0 -241
- package/test/TestDataHookFuzzing.sol +0 -524
- package/test/TestDurationUnderflow.sol +0 -233
- package/test/TestFeeFreeCashOutBypass.sol +0 -949
- package/test/TestFeeProcessingFailure.sol +0 -218
- package/test/TestFees.sol +0 -619
- package/test/TestForwardedTokenConsumption.sol +0 -425
- package/test/TestInterfaceSupport.sol +0 -81
- package/test/TestJBERC20Inheritance.sol +0 -103
- package/test/TestL2SequencerPriceFeed.sol +0 -292
- package/test/TestLaunchProject.sol +0 -188
- package/test/TestMetaTx.sol +0 -217
- package/test/TestMetadataOffsetOverflow.sol +0 -179
- package/test/TestMetadataParserLib.sol +0 -471
- package/test/TestMigrationHeldFees.sol +0 -255
- package/test/TestMintTokensOf.sol +0 -185
- package/test/TestMultiTerminalSurplus.sol +0 -348
- package/test/TestMultiTokenSurplus.sol +0 -202
- package/test/TestMultipleAccessLimits.sol +0 -664
- package/test/TestPayBurnRedeemFlow.sol +0 -195
- package/test/TestPayHooks.sol +0 -209
- package/test/TestPermissions.sol +0 -324
- package/test/TestPermissionsEdge.sol +0 -290
- package/test/TestPermit2DataHook.t.sol +0 -360
- package/test/TestPermit2Terminal.sol +0 -372
- package/test/TestRulesetQueueing.sol +0 -1025
- package/test/TestRulesetQueuingStress.sol +0 -806
- package/test/TestRulesetWeightCaching.sol +0 -178
- package/test/TestSplits.sol +0 -391
- package/test/TestTerminalMigration.sol +0 -274
- package/test/TestTerminalPreviewParity.sol +0 -208
- package/test/TestTokenFlow.sol +0 -191
- package/test/TestWeightCacheStaleAfterRejection.sol +0 -303
- package/test/WeirdTokenTests.t.sol +0 -817
- package/test/audit/CashOutReenterPay.t.sol +0 -501
- package/test/audit/CodexHeldFeeRounding.t.sol +0 -159
- package/test/audit/CodexMigrationFeeFailure.t.sol +0 -163
- package/test/audit/CrossTerminalSurplusSpoof.t.sol +0 -140
- package/test/audit/CycledSurplusAllowanceReset.t.sol +0 -184
- package/test/audit/FeeFreeSurplusLifecycle.t.sol +0 -399
- package/test/audit/FeeFreeSurplusStale.t.sol +0 -248
- package/test/audit/USDTVoidReturnCompat.t.sol +0 -525
- package/test/fork/TestChainlinkPriceFeedFork.sol +0 -254
- package/test/fork/TestSequencerPriceFeedFork.sol +0 -168
- package/test/fork/TestTerminalPreviewParityFork.sol +0 -108
- package/test/formal/BondingCurveProperties.t.sol +0 -420
- package/test/formal/FeeProperties.t.sol +0 -252
- package/test/invariants/Phase3DeepInvariant.t.sol +0 -412
- package/test/invariants/RulesetsInvariant.t.sol +0 -125
- package/test/invariants/TerminalStoreInvariant.t.sol +0 -227
- package/test/invariants/TokensInvariant.t.sol +0 -195
- package/test/invariants/handlers/ComprehensiveHandler.sol +0 -303
- package/test/invariants/handlers/EconomicHandler.sol +0 -377
- package/test/invariants/handlers/Phase3Handler.sol +0 -443
- package/test/invariants/handlers/RulesetsHandler.sol +0 -115
- package/test/invariants/handlers/TerminalStoreHandler.sol +0 -151
- package/test/invariants/handlers/TokensHandler.sol +0 -126
- package/test/regression/HoldFeesCashOutReserved.t.sol +0 -415
- package/test/regression/WeightCacheBoundary.t.sol +0 -291
- package/test/trees/JBController/burnTokensOf.tree +0 -9
- package/test/trees/JBController/claimTokensFor.tree +0 -5
- package/test/trees/JBController/deployERC20For.tree +0 -5
- package/test/trees/JBController/getRulesetOf.tree +0 -5
- package/test/trees/JBController/launchProjectFor.tree +0 -12
- package/test/trees/JBController/launchRulesetsFor.tree +0 -8
- package/test/trees/JBController/migrateController.tree +0 -12
- package/test/trees/JBController/mintTokensOf.tree +0 -12
- package/test/trees/JBController/payReservedTokenToTerminal.tree +0 -8
- package/test/trees/JBController/receiveMigrationFrom.tree +0 -4
- package/test/trees/JBController/sendReservedTokensToSplitsOf.tree +0 -12
- package/test/trees/JBController/setMetadataOf.tree +0 -5
- package/test/trees/JBController/setSplitGroupsOf.tree +0 -5
- package/test/trees/JBController/setTokenFor.tree +0 -5
- package/test/trees/JBController/transferCreditsFrom.tree +0 -8
- package/test/trees/JBDirectory/primaryTerminalOf.tree +0 -8
- package/test/trees/JBDirectory/setControllerOf.tree +0 -11
- package/test/trees/JBDirectory/setPrimaryTerminalOf.tree +0 -15
- package/test/trees/JBDirectory/setTerminalsOf.tree +0 -11
- package/test/trees/JBERC20/initialize.tree +0 -7
- package/test/trees/JBERC20/name.tree +0 -5
- package/test/trees/JBERC20/nonces.tree +0 -5
- package/test/trees/JBERC20/symbol.tree +0 -5
- package/test/trees/JBFeelessAddresses/setFeelessAddress.tree +0 -5
- package/test/trees/JBFeelessAddresses/supportsInterface.tree +0 -5
- package/test/trees/JBFundAccessLimits/payoutLimitOf.tree +0 -5
- package/test/trees/JBFundAccessLimits/payoutLimitsOf.tree +0 -8
- package/test/trees/JBFundAccessLimits/setFundAccessLimitsFor.tree +0 -18
- package/test/trees/JBFundAccessLimits/surplusAllowanceOf.tree +0 -5
- package/test/trees/JBFundAccessLimits/surplusAllowancesOf.tree +0 -8
- package/test/trees/JBMetadataResolver/getDataFor.tree +0 -8
- package/test/trees/JBMultiTerminal/accountingContextsOf.tree +0 -5
- package/test/trees/JBMultiTerminal/addAccountingContextsFor.tree +0 -10
- package/test/trees/JBMultiTerminal/addToBalanceOf.tree +0 -23
- package/test/trees/JBMultiTerminal/cashOutTokensOf.tree +0 -23
- package/test/trees/JBMultiTerminal/executePayout.tree +0 -32
- package/test/trees/JBMultiTerminal/executeProcessFee.tree +0 -14
- package/test/trees/JBMultiTerminal/migrateBalanceOf.tree +0 -12
- package/test/trees/JBMultiTerminal/pay.tree +0 -23
- package/test/trees/JBMultiTerminal/processHeldFeesOf.tree +0 -8
- package/test/trees/JBMultiTerminal/sendPayoutsOf.tree +0 -34
- package/test/trees/JBMultiTerminal/useAllowanceOf.tree +0 -16
- package/test/trees/JBPermissions/hasPermission.tree +0 -8
- package/test/trees/JBPermissions/hasPermissions.tree +0 -8
- package/test/trees/JBPermissions/setPermissionsFor.tree +0 -5
- package/test/trees/JBPrices/addPriceFeedFor.tree +0 -14
- package/test/trees/JBPrices/pricePerUnitOf.tree +0 -11
- package/test/trees/JBProjects/createFor.tree +0 -11
- package/test/trees/JBProjects/setTokenUriResolver.tree +0 -5
- package/test/trees/JBProjects/supportsInterface.tree +0 -9
- package/test/trees/JBProjects/tokenURI.tree +0 -5
- package/test/trees/JBRulesets/currentApprovalStatusForLatestRulesetOf.tree +0 -8
- package/test/trees/JBRulesets/currentOf.tree +0 -12
- package/test/trees/JBRulesets/getRulesetOf.tree +0 -5
- package/test/trees/JBRulesets/latestQueuedRulesetOf.tree +0 -10
- package/test/trees/JBRulesets/rulesetsOf.tree +0 -11
- package/test/trees/JBRulesets/upcomingRulesetOf.tree +0 -20
- package/test/trees/JBRulesets/updateRulesetWeightCache.tree +0 -5
- package/test/trees/JBSplits/setSplitGroupsOf.tree +0 -17
- package/test/trees/JBSplits/splitsOf.tree +0 -5
- package/test/trees/JBTerminalStore/currentReclaimableSurplusOf.tree +0 -16
- package/test/trees/JBTerminalStore/currentSurplusOf.tree +0 -25
- package/test/trees/JBTerminalStore/currentTotalSurplusOf.tree +0 -5
- package/test/trees/JBTerminalStore/recordCashOutsFor.tree +0 -16
- package/test/trees/JBTerminalStore/recordPaymentFrom.tree +0 -14
- package/test/trees/JBTerminalStore/recordPayoutFor.tree +0 -10
- package/test/trees/JBTerminalStore/recordTerminalMigration.tree +0 -5
- package/test/trees/JBTerminalStore/recordUsedAllowanceOf.tree +0 -10
- package/test/trees/JBTokens/burnFrom.tree +0 -10
- package/test/trees/JBTokens/claimTokensFor.tree +0 -10
- package/test/trees/JBTokens/deployERC20For.tree +0 -12
- package/test/trees/JBTokens/mintFor.tree +0 -10
- package/test/trees/JBTokens/setTokenFor.tree +0 -11
- package/test/trees/JBTokens/totalBalanceOf.tree +0 -5
- package/test/trees/JBTokens/totalSupplyOf.tree +0 -5
- package/test/trees/JBTokens/transferCreditsFrom.tree +0 -8
- package/test/trees/mintTokensOf.tree +0 -12
- package/test/units/static/JBChainlinkV3PriceFeed/TestPriceFeed.sol +0 -223
- package/test/units/static/JBController/JBControllerSetup.sol +0 -50
- package/test/units/static/JBController/TestBurnTokensOf.sol +0 -114
- package/test/units/static/JBController/TestClaimTokensFor.sol +0 -63
- package/test/units/static/JBController/TestDeployErc20For.sol +0 -86
- package/test/units/static/JBController/TestLaunchProjectFor.sol +0 -302
- package/test/units/static/JBController/TestLaunchRulesetsFor.sol +0 -342
- package/test/units/static/JBController/TestMigrateController.sol +0 -157
- package/test/units/static/JBController/TestMintTokensOfUnits.sol +0 -111
- package/test/units/static/JBController/TestOmnichainRulesetOperator.sol +0 -324
- package/test/units/static/JBController/TestPayReservedTokenToTerminal.sol +0 -74
- package/test/units/static/JBController/TestPreviewMintOf.sol +0 -117
- package/test/units/static/JBController/TestReceiveMigrationFrom.sol +0 -99
- package/test/units/static/JBController/TestRulesetViews.sol +0 -225
- package/test/units/static/JBController/TestSendReservedTokensToSplitsOf.sol +0 -615
- package/test/units/static/JBController/TestSetSplitGroupsOf.sol +0 -68
- package/test/units/static/JBController/TestSetTokenFor.sol +0 -239
- package/test/units/static/JBController/TestSetUriOf.sol +0 -57
- package/test/units/static/JBController/TestTransferCreditsFrom.sol +0 -169
- package/test/units/static/JBDeadline/TestDeadlineFuzz.sol +0 -211
- package/test/units/static/JBDirectory/JBDirectorySetup.sol +0 -26
- package/test/units/static/JBDirectory/TestPrimaryTerminalOf.sol +0 -126
- package/test/units/static/JBDirectory/TestSetControllerOf.sol +0 -183
- package/test/units/static/JBDirectory/TestSetControllerOfMigrationOrder.sol +0 -104
- package/test/units/static/JBDirectory/TestSetPrimaryTerminalOf.sol +0 -179
- package/test/units/static/JBDirectory/TestSetTerminalsOf.sol +0 -137
- package/test/units/static/JBERC20/JBERC20Setup.sol +0 -34
- package/test/units/static/JBERC20/SigUtils.sol +0 -36
- package/test/units/static/JBERC20/TestInitialize.sol +0 -60
- package/test/units/static/JBERC20/TestName.sol +0 -30
- package/test/units/static/JBERC20/TestNonces.sol +0 -62
- package/test/units/static/JBERC20/TestSymbol.sol +0 -31
- package/test/units/static/JBFeelessAdresses/JBFeelessSetup.sol +0 -22
- package/test/units/static/JBFeelessAdresses/TestInterfaces.sol +0 -30
- package/test/units/static/JBFeelessAdresses/TestSetFeelessAddress.sol +0 -35
- package/test/units/static/JBFees/TestFeesFuzz.sol +0 -79
- package/test/units/static/JBFixedPointNumber/TestAdjustDecimals.sol +0 -16
- package/test/units/static/JBFixedPointNumber/TestAdjustDecimalsFuzz.sol +0 -71
- package/test/units/static/JBFundAccessLimits/JBFundAccessSetup.sol +0 -24
- package/test/units/static/JBFundAccessLimits/TestFundAccessLimitsEdge.sol +0 -163
- package/test/units/static/JBFundAccessLimits/TestPayoutLimitOf.sol +0 -59
- package/test/units/static/JBFundAccessLimits/TestPayoutLimitsOf.sol +0 -101
- package/test/units/static/JBFundAccessLimits/TestSetFundAccessLimitsFor.sol +0 -189
- package/test/units/static/JBFundAccessLimits/TestSurplusAllowanceOf.sol +0 -64
- package/test/units/static/JBFundAccessLimits/TestSurplusAllowancesOf.sol +0 -102
- package/test/units/static/JBMetadataResolver/TestGetDataFor.sol +0 -90
- package/test/units/static/JBMetadataResolver/TestMetadataResolverEdgeCases.sol +0 -247
- package/test/units/static/JBMetadataResolver/TestMetadataResolverFuzz.sol +0 -229
- package/test/units/static/JBMultiTerminal/JBMultiTerminalSetup.sol +0 -50
- package/test/units/static/JBMultiTerminal/TestAccountingContextsOf.sol +0 -72
- package/test/units/static/JBMultiTerminal/TestAddAccountingContextsFor.sol +0 -289
- package/test/units/static/JBMultiTerminal/TestAddToBalanceOf.sol +0 -474
- package/test/units/static/JBMultiTerminal/TestCashOutTokensOf.sol +0 -624
- package/test/units/static/JBMultiTerminal/TestExecutePayout.sol +0 -578
- package/test/units/static/JBMultiTerminal/TestExecuteProcessFee.sol +0 -202
- package/test/units/static/JBMultiTerminal/TestMigrateBalanceOf.sol +0 -222
- package/test/units/static/JBMultiTerminal/TestPay.sol +0 -604
- package/test/units/static/JBMultiTerminal/TestPreviewCashOutFrom.sol +0 -117
- package/test/units/static/JBMultiTerminal/TestPreviewPayFor.sol +0 -114
- package/test/units/static/JBMultiTerminal/TestProcessHeldFeesOf.sol +0 -228
- package/test/units/static/JBMultiTerminal/TestSelfPayRevert.sol +0 -55
- package/test/units/static/JBMultiTerminal/TestSendPayoutsOf.sol +0 -257
- package/test/units/static/JBMultiTerminal/TestUseAllowanceOf.sol +0 -611
- package/test/units/static/JBPermissions/JBPermissionsSetup.sol +0 -20
- package/test/units/static/JBPermissions/TestHasPermission.sol +0 -50
- package/test/units/static/JBPermissions/TestHasPermissions.sol +0 -93
- package/test/units/static/JBPermissions/TestSetPermissionsFor.sol +0 -64
- package/test/units/static/JBPrices/JBPricesSetup.sol +0 -32
- package/test/units/static/JBPrices/TestAddPriceFeedFor.sol +0 -107
- package/test/units/static/JBPrices/TestPricePerUnitOf.sol +0 -132
- package/test/units/static/JBPrices/TestPrices.sol +0 -265
- package/test/units/static/JBProjects/JBProjectsSetup.sol +0 -22
- package/test/units/static/JBProjects/TestCreateFor.sol +0 -71
- package/test/units/static/JBProjects/TestInitialProject.sol +0 -21
- package/test/units/static/JBProjects/TestInterfaces.sol +0 -26
- package/test/units/static/JBProjects/TestSetResolver.sol +0 -37
- package/test/units/static/JBProjects/TestTokenUri.sol +0 -40
- package/test/units/static/JBRulesetMetadataResolver/TestSetCashOutTaxRateTo.sol +0 -108
- package/test/units/static/JBRulesets/JBRulesetsSetup.sol +0 -24
- package/test/units/static/JBRulesets/TestCurrentApprovalStatusForLatestRulesetOf.sol +0 -265
- package/test/units/static/JBRulesets/TestCurrentOf.sol +0 -242
- package/test/units/static/JBRulesets/TestGetRulesetOf.sol +0 -100
- package/test/units/static/JBRulesets/TestLatestQueuedRulesetOf.sol +0 -260
- package/test/units/static/JBRulesets/TestRulesets.sol +0 -632
- package/test/units/static/JBRulesets/TestRulesetsOf.sol +0 -37
- package/test/units/static/JBRulesets/TestUpcomingRulesetOf.sol +0 -522
- package/test/units/static/JBRulesets/TestUpdateRulesetWeightCache.sol +0 -96
- package/test/units/static/JBSplits/JBSplitsSetup.sol +0 -26
- package/test/units/static/JBSplits/TestSelfManagedSplitGroups.sol +0 -552
- package/test/units/static/JBSplits/TestSetSplitGroupsOf.sol +0 -377
- package/test/units/static/JBSplits/TestSplitsLockedEdge.sol +0 -267
- package/test/units/static/JBSplits/TestSplitsOf.sol +0 -24
- package/test/units/static/JBSplits/TestSplitsPacking.sol +0 -36
- package/test/units/static/JBSurplus/TestSurplusFuzz.sol +0 -160
- package/test/units/static/JBTerminalStore/JBTerminalStoreSetup.sol +0 -45
- package/test/units/static/JBTerminalStore/TestCurrentReclaimableSurplusOf.sol +0 -536
- package/test/units/static/JBTerminalStore/TestCurrentSurplusOf.sol +0 -463
- package/test/units/static/JBTerminalStore/TestCurrentTotalSurplusOf.sol +0 -135
- package/test/units/static/JBTerminalStore/TestPreviewCashOutFrom.sol +0 -476
- package/test/units/static/JBTerminalStore/TestPreviewPayFrom.sol +0 -494
- package/test/units/static/JBTerminalStore/TestRecordCashOutsFor.sol +0 -652
- package/test/units/static/JBTerminalStore/TestRecordPaymentFrom.sol +0 -744
- package/test/units/static/JBTerminalStore/TestRecordPayoutFor.sol +0 -289
- package/test/units/static/JBTerminalStore/TestRecordTerminalMigration.sol +0 -138
- package/test/units/static/JBTerminalStore/TestRecordUsedAllowanceOf.sol +0 -415
- package/test/units/static/JBTerminalStore/TestUint224Overflow.sol +0 -219
- package/test/units/static/JBTokens/JBTokensSetup.sol +0 -32
- package/test/units/static/JBTokens/TestBurnFrom.sol +0 -107
- package/test/units/static/JBTokens/TestClaimTokensFor.sol +0 -110
- package/test/units/static/JBTokens/TestDeployERC20ForUnits.sol +0 -92
- package/test/units/static/JBTokens/TestMintFor.sol +0 -100
- package/test/units/static/JBTokens/TestSetTokenFor.sol +0 -98
- package/test/units/static/JBTokens/TestTotalBalanceOf.sol +0 -65
- package/test/units/static/JBTokens/TestTotalSupplyOf.sol +0 -56
- package/test/units/static/JBTokens/TestTransferCreditsFrom.sol +0 -56
|
@@ -1,348 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: MIT
|
|
2
|
-
pragma solidity ^0.8.6;
|
|
3
|
-
|
|
4
|
-
import {TestBaseWorkflow} from "./helpers/TestBaseWorkflow.sol";
|
|
5
|
-
import {JBMultiTerminal} from "../src/JBMultiTerminal.sol";
|
|
6
|
-
import {IJBController} from "../src/interfaces/IJBController.sol";
|
|
7
|
-
import {IJBRulesetApprovalHook} from "../src/interfaces/IJBRulesetApprovalHook.sol";
|
|
8
|
-
import {IJBTokens} from "../src/interfaces/IJBTokens.sol";
|
|
9
|
-
import {JBConstants} from "../src/libraries/JBConstants.sol";
|
|
10
|
-
import {JBAccountingContext} from "../src/structs/JBAccountingContext.sol";
|
|
11
|
-
import {JBFundAccessLimitGroup} from "../src/structs/JBFundAccessLimitGroup.sol";
|
|
12
|
-
import {JBRulesetConfig} from "../src/structs/JBRulesetConfig.sol";
|
|
13
|
-
import {JBRulesetMetadata} from "../src/structs/JBRulesetMetadata.sol";
|
|
14
|
-
import {JBSplitGroup} from "../src/structs/JBSplitGroup.sol";
|
|
15
|
-
import {JBTerminalConfig} from "../src/structs/JBTerminalConfig.sol";
|
|
16
|
-
import {MockPriceFeed} from "./mock/MockPriceFeed.sol";
|
|
17
|
-
import {MockERC20} from "./mock/MockERC20.sol";
|
|
18
|
-
|
|
19
|
-
/// @notice Tests for multi-terminal surplus aggregation edge cases, including cross-terminal
|
|
20
|
-
/// surplus with useTotalSurplusForCashOuts, cash out balance limits, and price conversion.
|
|
21
|
-
contract TestMultiTerminalSurplus_Local is TestBaseWorkflow {
|
|
22
|
-
IJBController private _controller;
|
|
23
|
-
JBMultiTerminal private _terminal1;
|
|
24
|
-
JBMultiTerminal private _terminal2;
|
|
25
|
-
IJBTokens private _tokens;
|
|
26
|
-
address private _projectOwner;
|
|
27
|
-
address private _user;
|
|
28
|
-
|
|
29
|
-
MockPriceFeed private _ethToUsdcFeed;
|
|
30
|
-
MockERC20 private _usdc;
|
|
31
|
-
|
|
32
|
-
uint32 private _nativeCurrency;
|
|
33
|
-
uint32 private _usdcCurrency;
|
|
34
|
-
|
|
35
|
-
uint256 private _projectId;
|
|
36
|
-
|
|
37
|
-
function setUp() public override {
|
|
38
|
-
super.setUp();
|
|
39
|
-
|
|
40
|
-
_projectOwner = multisig();
|
|
41
|
-
_user = beneficiary();
|
|
42
|
-
_controller = jbController();
|
|
43
|
-
_terminal1 = jbMultiTerminal();
|
|
44
|
-
_terminal2 = jbMultiTerminal2();
|
|
45
|
-
_tokens = jbTokens();
|
|
46
|
-
_usdc = usdcToken();
|
|
47
|
-
|
|
48
|
-
_nativeCurrency = uint32(uint160(JBConstants.NATIVE_TOKEN));
|
|
49
|
-
_usdcCurrency = uint32(uint160(address(_usdc)));
|
|
50
|
-
|
|
51
|
-
// Price feed: 1 ETH = 2000 USDC (6 decimals).
|
|
52
|
-
_ethToUsdcFeed = new MockPriceFeed(2000e6, 6);
|
|
53
|
-
|
|
54
|
-
// Metadata with useTotalSurplusForCashOuts = true.
|
|
55
|
-
JBRulesetMetadata memory _metadata = JBRulesetMetadata({
|
|
56
|
-
reservedPercent: 0,
|
|
57
|
-
cashOutTaxRate: 0,
|
|
58
|
-
baseCurrency: _nativeCurrency,
|
|
59
|
-
pausePay: false,
|
|
60
|
-
pauseCreditTransfers: false,
|
|
61
|
-
allowOwnerMinting: true,
|
|
62
|
-
allowSetCustomToken: true,
|
|
63
|
-
allowTerminalMigration: false,
|
|
64
|
-
allowSetTerminals: false,
|
|
65
|
-
allowSetController: false,
|
|
66
|
-
allowAddAccountingContext: true,
|
|
67
|
-
allowAddPriceFeed: true,
|
|
68
|
-
ownerMustSendPayouts: false,
|
|
69
|
-
holdFees: false,
|
|
70
|
-
useTotalSurplusForCashOuts: true,
|
|
71
|
-
useDataHookForPay: false,
|
|
72
|
-
useDataHookForCashOut: false,
|
|
73
|
-
dataHook: address(0),
|
|
74
|
-
metadata: 0
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
JBRulesetConfig[] memory _rulesetConfig = new JBRulesetConfig[](1);
|
|
78
|
-
_rulesetConfig[0].mustStartAtOrAfter = 0;
|
|
79
|
-
_rulesetConfig[0].duration = 0;
|
|
80
|
-
_rulesetConfig[0].weight = 1000 * 10 ** 18;
|
|
81
|
-
_rulesetConfig[0].weightCutPercent = 0;
|
|
82
|
-
_rulesetConfig[0].approvalHook = IJBRulesetApprovalHook(address(0));
|
|
83
|
-
_rulesetConfig[0].metadata = _metadata;
|
|
84
|
-
_rulesetConfig[0].splitGroups = new JBSplitGroup[](0);
|
|
85
|
-
_rulesetConfig[0].fundAccessLimitGroups = new JBFundAccessLimitGroup[](0);
|
|
86
|
-
|
|
87
|
-
// Terminal 1 accepts ETH; Terminal 2 accepts USDC.
|
|
88
|
-
JBAccountingContext[] memory _ethContext = new JBAccountingContext[](1);
|
|
89
|
-
_ethContext[0] = JBAccountingContext({token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: _nativeCurrency});
|
|
90
|
-
|
|
91
|
-
JBAccountingContext[] memory _usdcContext = new JBAccountingContext[](1);
|
|
92
|
-
_usdcContext[0] = JBAccountingContext({token: address(_usdc), decimals: 6, currency: _usdcCurrency});
|
|
93
|
-
|
|
94
|
-
JBTerminalConfig[] memory _terminalConfigs = new JBTerminalConfig[](2);
|
|
95
|
-
_terminalConfigs[0] = JBTerminalConfig({terminal: _terminal1, accountingContextsToAccept: _ethContext});
|
|
96
|
-
_terminalConfigs[1] = JBTerminalConfig({terminal: _terminal2, accountingContextsToAccept: _usdcContext});
|
|
97
|
-
|
|
98
|
-
// Launch project with both terminals.
|
|
99
|
-
_projectId = _controller.launchProjectFor({
|
|
100
|
-
owner: _projectOwner,
|
|
101
|
-
projectUri: "multi-terminal-surplus-test",
|
|
102
|
-
rulesetConfigurations: _rulesetConfig,
|
|
103
|
-
terminalConfigurations: _terminalConfigs,
|
|
104
|
-
memo: ""
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
// Add price feed: USDC priced in native token terms.
|
|
108
|
-
vm.prank(_projectOwner);
|
|
109
|
-
_controller.addPriceFeedFor({
|
|
110
|
-
projectId: _projectId, pricingCurrency: _usdcCurrency, unitCurrency: _nativeCurrency, feed: _ethToUsdcFeed
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
// Deploy ERC-20 for the project.
|
|
114
|
-
vm.prank(_projectOwner);
|
|
115
|
-
_controller.deployERC20For(_projectId, "MultiTermToken", "MTT", bytes32(0));
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
/// @notice Helper: pay ETH into terminal 1.
|
|
119
|
-
function _payEth(address payer, uint256 amount) internal returns (uint256 tokensReceived) {
|
|
120
|
-
vm.deal(payer, amount);
|
|
121
|
-
vm.prank(payer);
|
|
122
|
-
tokensReceived = _terminal1.pay{value: amount}({
|
|
123
|
-
projectId: _projectId,
|
|
124
|
-
amount: amount,
|
|
125
|
-
token: JBConstants.NATIVE_TOKEN,
|
|
126
|
-
beneficiary: payer,
|
|
127
|
-
minReturnedTokens: 0,
|
|
128
|
-
memo: "",
|
|
129
|
-
metadata: ""
|
|
130
|
-
});
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
/// @notice Helper: pay USDC into terminal 2.
|
|
134
|
-
function _payUsdc(address payer, uint256 amount) internal returns (uint256 tokensReceived) {
|
|
135
|
-
_usdc.mint(payer, amount);
|
|
136
|
-
vm.prank(payer);
|
|
137
|
-
_usdc.approve(address(permit2()), amount);
|
|
138
|
-
vm.prank(payer);
|
|
139
|
-
// forge-lint: disable-next-line(unsafe-typecast)
|
|
140
|
-
permit2().approve(address(_usdc), address(_terminal2), uint160(amount), type(uint48).max);
|
|
141
|
-
|
|
142
|
-
vm.prank(payer);
|
|
143
|
-
tokensReceived = _terminal2.pay({
|
|
144
|
-
projectId: _projectId,
|
|
145
|
-
amount: amount,
|
|
146
|
-
token: address(_usdc),
|
|
147
|
-
beneficiary: payer,
|
|
148
|
-
minReturnedTokens: 0,
|
|
149
|
-
memo: "",
|
|
150
|
-
metadata: ""
|
|
151
|
-
});
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
/// @notice Surplus aggregation across 2 terminals with different tokens.
|
|
155
|
-
function test_surplusAcrossTwoTerminals() public {
|
|
156
|
-
uint256 ethAmount = 2 ether;
|
|
157
|
-
uint256 usdcAmount = 4000e6; // $4000 = 2 ETH
|
|
158
|
-
|
|
159
|
-
_payEth(_user, ethAmount);
|
|
160
|
-
_payUsdc(_user, usdcAmount);
|
|
161
|
-
|
|
162
|
-
// Check ETH-only surplus from terminal 1.
|
|
163
|
-
JBAccountingContext[] memory ethCtx = new JBAccountingContext[](1);
|
|
164
|
-
ethCtx[0] = JBAccountingContext({token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: _nativeCurrency});
|
|
165
|
-
uint256 ethSurplus = _terminal1.currentSurplusOf(_projectId, new address[](0), 18, _nativeCurrency);
|
|
166
|
-
assertEq(ethSurplus, ethAmount, "terminal1 ETH surplus should match payment");
|
|
167
|
-
|
|
168
|
-
// Check USDC-only surplus from terminal 2.
|
|
169
|
-
JBAccountingContext[] memory usdcCtx = new JBAccountingContext[](1);
|
|
170
|
-
usdcCtx[0] = JBAccountingContext({token: address(_usdc), decimals: 6, currency: _usdcCurrency});
|
|
171
|
-
uint256 usdcSurplus = _terminal2.currentSurplusOf(_projectId, new address[](0), 6, _usdcCurrency);
|
|
172
|
-
assertEq(usdcSurplus, usdcAmount, "terminal2 USDC surplus should match payment");
|
|
173
|
-
|
|
174
|
-
// Check total surplus in ETH terms from the store.
|
|
175
|
-
uint256 totalSurplusEth = jbTerminalStore().currentTotalSurplusOf(_projectId, 18, _nativeCurrency);
|
|
176
|
-
// Should be ETH amount + USDC-converted-to-ETH.
|
|
177
|
-
// The total should be greater than just the ETH amount.
|
|
178
|
-
assertGt(totalSurplusEth, ethAmount, "total surplus should include converted USDC value");
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
/// @notice With useTotalSurplusForCashOuts enabled, cash out from terminal 1 uses combined surplus.
|
|
182
|
-
function test_cashOutUsesTotalSurplusWhenEnabled() public {
|
|
183
|
-
uint256 ethAmount = 1 ether;
|
|
184
|
-
uint256 usdcAmount = 2000e6; // $2000 = 1 ETH
|
|
185
|
-
|
|
186
|
-
uint256 userEthTokens = _payEth(_user, ethAmount);
|
|
187
|
-
_payUsdc(_user, usdcAmount);
|
|
188
|
-
|
|
189
|
-
// User cashes out from terminal 1 (ETH).
|
|
190
|
-
// useTotalSurplusForCashOuts is enabled, so the bonding curve should consider
|
|
191
|
-
// the total surplus across both terminals, not just terminal 1's ETH balance.
|
|
192
|
-
uint256 userBalanceBefore = _user.balance;
|
|
193
|
-
|
|
194
|
-
vm.prank(_user);
|
|
195
|
-
uint256 reclaimed = _terminal1.cashOutTokensOf({
|
|
196
|
-
holder: _user,
|
|
197
|
-
projectId: _projectId,
|
|
198
|
-
cashOutCount: userEthTokens / 4, // cash out 25% of tokens from ETH payment
|
|
199
|
-
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
200
|
-
minTokensReclaimed: 0,
|
|
201
|
-
beneficiary: payable(_user),
|
|
202
|
-
metadata: new bytes(0)
|
|
203
|
-
});
|
|
204
|
-
|
|
205
|
-
// Reclaim amount should be > 0.
|
|
206
|
-
assertGt(reclaimed, 0, "should reclaim some ETH");
|
|
207
|
-
|
|
208
|
-
// Verify the user received ETH.
|
|
209
|
-
assertGt(_user.balance, userBalanceBefore, "user balance should increase");
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
/// @notice Cash out from terminal 1 cannot extract more ETH than terminal 1 holds,
|
|
213
|
-
/// even with useTotalSurplusForCashOuts enabled.
|
|
214
|
-
function test_cashOutCannotExceedTerminalBalance() public {
|
|
215
|
-
uint256 ethAmount = 1 ether;
|
|
216
|
-
uint256 usdcAmount = 20_000e6; // $20,000 = 10 ETH -- much more value than in terminal 1
|
|
217
|
-
|
|
218
|
-
uint256 userEthTokens = _payEth(_user, ethAmount);
|
|
219
|
-
_payUsdc(_user, usdcAmount);
|
|
220
|
-
|
|
221
|
-
// The total surplus in ETH terms is ~11 ETH, but terminal 1 only holds 1 ETH.
|
|
222
|
-
// Cashing out all tokens from terminal 1 should be bounded by the terminal's ETH balance.
|
|
223
|
-
// With cashOutTaxRate=0 and useTotalSurplusForCashOuts=true, the reclaim amount
|
|
224
|
-
// = totalSurplus * cashOutCount / totalSupply.
|
|
225
|
-
// The total token supply includes tokens from both payments.
|
|
226
|
-
|
|
227
|
-
// Get total token balance.
|
|
228
|
-
uint256 totalTokenBalance = _tokens.totalBalanceOf(_user, _projectId);
|
|
229
|
-
assertGt(totalTokenBalance, userEthTokens, "total tokens should include USDC payment tokens");
|
|
230
|
-
|
|
231
|
-
// Cash out a large fraction -- the reclaim may be limited by what terminal 1 actually holds.
|
|
232
|
-
// If reclaimAmount + hookAmounts > terminal balance, the store should revert.
|
|
233
|
-
// Let us try cashing out enough tokens that the calculated reclaim exceeds terminal 1's balance.
|
|
234
|
-
// With 1 ETH in terminal 1 and ~11 ETH total surplus, cashing out all tokens
|
|
235
|
-
// would try to reclaim ~11 ETH from a terminal that only has 1 ETH.
|
|
236
|
-
|
|
237
|
-
vm.prank(_user);
|
|
238
|
-
vm.expectRevert(); // Should revert with InadequateTerminalStoreBalance
|
|
239
|
-
_terminal1.cashOutTokensOf({
|
|
240
|
-
holder: _user,
|
|
241
|
-
projectId: _projectId,
|
|
242
|
-
cashOutCount: totalTokenBalance,
|
|
243
|
-
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
244
|
-
minTokensReclaimed: 0,
|
|
245
|
-
beneficiary: payable(_user),
|
|
246
|
-
metadata: new bytes(0)
|
|
247
|
-
});
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
/// @notice Surplus aggregation with price conversion rounding: small amounts.
|
|
251
|
-
function test_surplusAggregationRounding() public {
|
|
252
|
-
// Pay a very small amount of ETH.
|
|
253
|
-
uint256 ethAmount = 1 wei;
|
|
254
|
-
_payEth(_user, ethAmount);
|
|
255
|
-
|
|
256
|
-
// Pay a very small amount of USDC.
|
|
257
|
-
uint256 usdcAmount = 1; // 1e-6 USDC = $0.000001
|
|
258
|
-
_payUsdc(_user, usdcAmount);
|
|
259
|
-
|
|
260
|
-
// Total surplus should be >= the ETH amount (even if USDC rounds to 0 when converted).
|
|
261
|
-
uint256 totalSurplus = jbTerminalStore().currentTotalSurplusOf(_projectId, 18, _nativeCurrency);
|
|
262
|
-
assertGe(totalSurplus, ethAmount, "total surplus should be at least the ETH amount");
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
/// @notice Per-terminal balance tracking is independent across terminals.
|
|
266
|
-
function test_perTerminalBalanceIndependence() public {
|
|
267
|
-
uint256 ethAmount = 5 ether;
|
|
268
|
-
uint256 usdcAmount = 3000e6;
|
|
269
|
-
|
|
270
|
-
_payEth(_user, ethAmount);
|
|
271
|
-
_payUsdc(_user, usdcAmount);
|
|
272
|
-
|
|
273
|
-
// Terminal 1 balance (ETH) should be unaffected by terminal 2's USDC.
|
|
274
|
-
uint256 t1EthBalance = jbTerminalStore().balanceOf(address(_terminal1), _projectId, JBConstants.NATIVE_TOKEN);
|
|
275
|
-
assertEq(t1EthBalance, ethAmount, "terminal 1 ETH balance should match");
|
|
276
|
-
|
|
277
|
-
// Terminal 2 balance (USDC) should be unaffected by terminal 1's ETH.
|
|
278
|
-
uint256 t2UsdcBalance = jbTerminalStore().balanceOf(address(_terminal2), _projectId, address(_usdc));
|
|
279
|
-
assertEq(t2UsdcBalance, usdcAmount, "terminal 2 USDC balance should match");
|
|
280
|
-
|
|
281
|
-
// Terminal 1 should have no USDC balance.
|
|
282
|
-
uint256 t1UsdcBalance = jbTerminalStore().balanceOf(address(_terminal1), _projectId, address(_usdc));
|
|
283
|
-
assertEq(t1UsdcBalance, 0, "terminal 1 should have no USDC balance");
|
|
284
|
-
|
|
285
|
-
// Terminal 2 should have no ETH balance.
|
|
286
|
-
uint256 t2EthBalance = jbTerminalStore().balanceOf(address(_terminal2), _projectId, JBConstants.NATIVE_TOKEN);
|
|
287
|
-
assertEq(t2EthBalance, 0, "terminal 2 should have no ETH balance");
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
/// @notice Small cash out from terminal with most funds -- verifiable amount.
|
|
291
|
-
function test_smallCashOutFromTerminalWithFunds() public {
|
|
292
|
-
uint256 ethAmount = 10 ether;
|
|
293
|
-
uint256 usdcAmount = 2000e6; // = 1 ETH equivalent
|
|
294
|
-
|
|
295
|
-
uint256 userTokens = _payEth(_user, ethAmount);
|
|
296
|
-
_payUsdc(_user, usdcAmount);
|
|
297
|
-
|
|
298
|
-
// Cash out a small fraction of tokens from the ETH terminal.
|
|
299
|
-
uint256 cashOutCount = userTokens / 100; // 1% of ETH-minted tokens
|
|
300
|
-
|
|
301
|
-
uint256 balanceBefore = _user.balance;
|
|
302
|
-
vm.prank(_user);
|
|
303
|
-
uint256 reclaimed = _terminal1.cashOutTokensOf({
|
|
304
|
-
holder: _user,
|
|
305
|
-
projectId: _projectId,
|
|
306
|
-
cashOutCount: cashOutCount,
|
|
307
|
-
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
308
|
-
minTokensReclaimed: 0,
|
|
309
|
-
beneficiary: payable(_user),
|
|
310
|
-
metadata: new bytes(0)
|
|
311
|
-
});
|
|
312
|
-
|
|
313
|
-
assertGt(reclaimed, 0, "should reclaim something for 1% cash out");
|
|
314
|
-
assertEq(_user.balance - balanceBefore, reclaimed, "user balance increase should match reclaimed amount");
|
|
315
|
-
|
|
316
|
-
// After small cash out, terminal 1 should still have most of its ETH.
|
|
317
|
-
uint256 remainingBalance =
|
|
318
|
-
jbTerminalStore().balanceOf(address(_terminal1), _projectId, JBConstants.NATIVE_TOKEN);
|
|
319
|
-
assertGt(remainingBalance, 9 ether, "most of the ETH should remain");
|
|
320
|
-
}
|
|
321
|
-
|
|
322
|
-
/// @notice Total surplus query from terminal store matches sum of individual terminal surpluses.
|
|
323
|
-
function test_totalSurplusConsistency() public {
|
|
324
|
-
uint256 ethAmount = 3 ether;
|
|
325
|
-
uint256 usdcAmount = 6000e6; // $6000 = 3 ETH
|
|
326
|
-
|
|
327
|
-
_payEth(_user, ethAmount);
|
|
328
|
-
_payUsdc(_user, usdcAmount);
|
|
329
|
-
|
|
330
|
-
// Get individual surpluses in ETH terms.
|
|
331
|
-
JBAccountingContext[] memory ethCtx = new JBAccountingContext[](1);
|
|
332
|
-
ethCtx[0] = JBAccountingContext({token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: _nativeCurrency});
|
|
333
|
-
uint256 t1Surplus = _terminal1.currentSurplusOf(_projectId, new address[](0), 18, _nativeCurrency);
|
|
334
|
-
|
|
335
|
-
JBAccountingContext[] memory usdcCtx = new JBAccountingContext[](1);
|
|
336
|
-
usdcCtx[0] = JBAccountingContext({token: address(_usdc), decimals: 6, currency: _usdcCurrency});
|
|
337
|
-
// Get terminal 2 surplus in ETH terms.
|
|
338
|
-
uint256 t2SurplusInEth = _terminal2.currentSurplusOf(_projectId, new address[](0), 18, _nativeCurrency);
|
|
339
|
-
|
|
340
|
-
// Get total surplus from the store.
|
|
341
|
-
uint256 totalSurplus = jbTerminalStore().currentTotalSurplusOf(_projectId, 18, _nativeCurrency);
|
|
342
|
-
|
|
343
|
-
// Total should equal sum of individual surpluses (both converted to ETH).
|
|
344
|
-
assertEq(totalSurplus, t1Surplus + t2SurplusInEth, "total surplus should equal sum of terminal surpluses");
|
|
345
|
-
}
|
|
346
|
-
|
|
347
|
-
receive() external payable {}
|
|
348
|
-
}
|
|
@@ -1,202 +0,0 @@
|
|
|
1
|
-
// SPDX-License-Identifier: MIT
|
|
2
|
-
pragma solidity ^0.8.6;
|
|
3
|
-
|
|
4
|
-
import {TestBaseWorkflow} from "./helpers/TestBaseWorkflow.sol";
|
|
5
|
-
import {JBMultiTerminal} from "../src/JBMultiTerminal.sol";
|
|
6
|
-
import {IJBController} from "../src/interfaces/IJBController.sol";
|
|
7
|
-
import {IJBRulesetApprovalHook} from "../src/interfaces/IJBRulesetApprovalHook.sol";
|
|
8
|
-
import {JBConstants} from "../src/libraries/JBConstants.sol";
|
|
9
|
-
import {JBAccountingContext} from "../src/structs/JBAccountingContext.sol";
|
|
10
|
-
import {JBFundAccessLimitGroup} from "../src/structs/JBFundAccessLimitGroup.sol";
|
|
11
|
-
import {JBRulesetConfig} from "../src/structs/JBRulesetConfig.sol";
|
|
12
|
-
import {JBRulesetMetadata} from "../src/structs/JBRulesetMetadata.sol";
|
|
13
|
-
import {JBSplitGroup} from "../src/structs/JBSplitGroup.sol";
|
|
14
|
-
import {JBTerminalConfig} from "../src/structs/JBTerminalConfig.sol";
|
|
15
|
-
import {MockPriceFeed} from "./mock/MockPriceFeed.sol";
|
|
16
|
-
import {MockERC20} from "./mock/MockERC20.sol";
|
|
17
|
-
|
|
18
|
-
/// @notice E2E test: ETH + USDC project with price feed, cross-token surplus aggregation.
|
|
19
|
-
contract TestMultiTokenSurplus_Local is TestBaseWorkflow {
|
|
20
|
-
IJBController private _controller;
|
|
21
|
-
JBMultiTerminal private _terminal;
|
|
22
|
-
uint256 private _projectId;
|
|
23
|
-
address private _projectOwner;
|
|
24
|
-
address private _beneficiary;
|
|
25
|
-
|
|
26
|
-
MockPriceFeed private _ethToUsdcFeed;
|
|
27
|
-
MockERC20 private _usdc;
|
|
28
|
-
|
|
29
|
-
uint32 private _nativeCurrency;
|
|
30
|
-
uint32 private _usdcCurrency;
|
|
31
|
-
|
|
32
|
-
function setUp() public override {
|
|
33
|
-
super.setUp();
|
|
34
|
-
|
|
35
|
-
_projectOwner = multisig();
|
|
36
|
-
_beneficiary = beneficiary();
|
|
37
|
-
_controller = jbController();
|
|
38
|
-
_terminal = jbMultiTerminal();
|
|
39
|
-
_usdc = usdcToken();
|
|
40
|
-
|
|
41
|
-
_nativeCurrency = uint32(uint160(JBConstants.NATIVE_TOKEN));
|
|
42
|
-
_usdcCurrency = uint32(uint160(address(_usdc)));
|
|
43
|
-
|
|
44
|
-
// Price feed: 1 ETH = 2000 USDC
|
|
45
|
-
// Feed reports price of 1 unit of native token in USDC terms
|
|
46
|
-
_ethToUsdcFeed = new MockPriceFeed(2000e6, 6);
|
|
47
|
-
|
|
48
|
-
JBRulesetMetadata memory _metadata = JBRulesetMetadata({
|
|
49
|
-
reservedPercent: 0,
|
|
50
|
-
cashOutTaxRate: 0,
|
|
51
|
-
baseCurrency: _nativeCurrency,
|
|
52
|
-
pausePay: false,
|
|
53
|
-
pauseCreditTransfers: false,
|
|
54
|
-
allowOwnerMinting: false,
|
|
55
|
-
allowSetCustomToken: false,
|
|
56
|
-
allowTerminalMigration: false,
|
|
57
|
-
allowSetTerminals: false,
|
|
58
|
-
allowSetController: false,
|
|
59
|
-
allowAddAccountingContext: true,
|
|
60
|
-
allowAddPriceFeed: true,
|
|
61
|
-
ownerMustSendPayouts: false,
|
|
62
|
-
holdFees: false,
|
|
63
|
-
useTotalSurplusForCashOuts: false,
|
|
64
|
-
useDataHookForPay: false,
|
|
65
|
-
useDataHookForCashOut: false,
|
|
66
|
-
dataHook: address(0),
|
|
67
|
-
metadata: 0
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
JBRulesetConfig[] memory _rulesetConfig = new JBRulesetConfig[](1);
|
|
71
|
-
_rulesetConfig[0].mustStartAtOrAfter = 0;
|
|
72
|
-
_rulesetConfig[0].duration = 0;
|
|
73
|
-
_rulesetConfig[0].weight = 1000 * 10 ** 18;
|
|
74
|
-
_rulesetConfig[0].weightCutPercent = 0;
|
|
75
|
-
_rulesetConfig[0].approvalHook = IJBRulesetApprovalHook(address(0));
|
|
76
|
-
_rulesetConfig[0].metadata = _metadata;
|
|
77
|
-
_rulesetConfig[0].splitGroups = new JBSplitGroup[](0);
|
|
78
|
-
_rulesetConfig[0].fundAccessLimitGroups = new JBFundAccessLimitGroup[](0);
|
|
79
|
-
|
|
80
|
-
// Accept both ETH and USDC
|
|
81
|
-
JBAccountingContext[] memory _tokensToAccept = new JBAccountingContext[](2);
|
|
82
|
-
_tokensToAccept[0] =
|
|
83
|
-
JBAccountingContext({token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: _nativeCurrency});
|
|
84
|
-
_tokensToAccept[1] = JBAccountingContext({token: address(_usdc), decimals: 6, currency: _usdcCurrency});
|
|
85
|
-
|
|
86
|
-
JBTerminalConfig[] memory _terminalConfigs = new JBTerminalConfig[](1);
|
|
87
|
-
_terminalConfigs[0] = JBTerminalConfig({terminal: _terminal, accountingContextsToAccept: _tokensToAccept});
|
|
88
|
-
|
|
89
|
-
_projectId = _controller.launchProjectFor({
|
|
90
|
-
owner: address(_projectOwner),
|
|
91
|
-
projectUri: "multi-token-surplus-test",
|
|
92
|
-
rulesetConfigurations: _rulesetConfig,
|
|
93
|
-
terminalConfigurations: _terminalConfigs,
|
|
94
|
-
memo: ""
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
// Add price feed: USDC priced in terms of native token
|
|
98
|
-
// This allows the terminal to convert USDC balances to ETH-denominated surplus
|
|
99
|
-
vm.prank(_projectOwner);
|
|
100
|
-
_controller.addPriceFeedFor({
|
|
101
|
-
projectId: _projectId, pricingCurrency: _usdcCurrency, unitCurrency: _nativeCurrency, feed: _ethToUsdcFeed
|
|
102
|
-
});
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
/// @notice ETH-only surplus is correct.
|
|
106
|
-
function test_ethOnlySurplus() public {
|
|
107
|
-
uint256 payAmount = 5 ether;
|
|
108
|
-
|
|
109
|
-
vm.deal(_beneficiary, payAmount);
|
|
110
|
-
vm.prank(_beneficiary);
|
|
111
|
-
_terminal.pay{value: payAmount}(_projectId, JBConstants.NATIVE_TOKEN, payAmount, _beneficiary, 0, "", "");
|
|
112
|
-
|
|
113
|
-
// Check surplus in ETH terms
|
|
114
|
-
JBAccountingContext[] memory contexts = new JBAccountingContext[](1);
|
|
115
|
-
contexts[0] = JBAccountingContext({token: JBConstants.NATIVE_TOKEN, decimals: 18, currency: _nativeCurrency});
|
|
116
|
-
|
|
117
|
-
uint256 surplus = _terminal.currentSurplusOf(_projectId, new address[](0), 18, _nativeCurrency);
|
|
118
|
-
assertEq(surplus, payAmount, "ETH surplus should match payment");
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
/// @notice USDC-only surplus is correct in USDC terms.
|
|
122
|
-
function test_usdcOnlySurplus() public {
|
|
123
|
-
uint256 usdcAmount = 2000e6; // $2000
|
|
124
|
-
|
|
125
|
-
// Mint USDC to beneficiary and approve
|
|
126
|
-
_usdc.mint(_beneficiary, usdcAmount);
|
|
127
|
-
vm.prank(_beneficiary);
|
|
128
|
-
_usdc.approve(address(permit2()), usdcAmount);
|
|
129
|
-
vm.prank(_beneficiary);
|
|
130
|
-
// forge-lint: disable-next-line(unsafe-typecast)
|
|
131
|
-
permit2().approve(address(_usdc), address(_terminal), uint160(usdcAmount), type(uint48).max);
|
|
132
|
-
|
|
133
|
-
vm.prank(_beneficiary);
|
|
134
|
-
_terminal.pay(_projectId, address(_usdc), usdcAmount, _beneficiary, 0, "", "");
|
|
135
|
-
|
|
136
|
-
// Check surplus in USDC terms
|
|
137
|
-
JBAccountingContext[] memory contexts = new JBAccountingContext[](1);
|
|
138
|
-
contexts[0] = JBAccountingContext({token: address(_usdc), decimals: 6, currency: _usdcCurrency});
|
|
139
|
-
|
|
140
|
-
uint256 surplus = _terminal.currentSurplusOf(_projectId, new address[](0), 6, _usdcCurrency);
|
|
141
|
-
assertEq(surplus, usdcAmount, "USDC surplus should match payment");
|
|
142
|
-
}
|
|
143
|
-
|
|
144
|
-
/// @notice Multi-token surplus aggregation converts USDC to ETH.
|
|
145
|
-
function test_multiTokenSurplus_aggregation() public {
|
|
146
|
-
uint256 ethAmount = 1 ether;
|
|
147
|
-
uint256 usdcAmount = 2000e6; // $2000 = 1 ETH at our price feed rate
|
|
148
|
-
|
|
149
|
-
// Pay ETH
|
|
150
|
-
vm.deal(_beneficiary, ethAmount);
|
|
151
|
-
vm.prank(_beneficiary);
|
|
152
|
-
_terminal.pay{value: ethAmount}(_projectId, JBConstants.NATIVE_TOKEN, ethAmount, _beneficiary, 0, "", "");
|
|
153
|
-
|
|
154
|
-
// Pay USDC
|
|
155
|
-
_usdc.mint(_beneficiary, usdcAmount);
|
|
156
|
-
vm.prank(_beneficiary);
|
|
157
|
-
_usdc.approve(address(permit2()), usdcAmount);
|
|
158
|
-
vm.prank(_beneficiary);
|
|
159
|
-
// forge-lint: disable-next-line(unsafe-typecast)
|
|
160
|
-
permit2().approve(address(_usdc), address(_terminal), uint160(usdcAmount), type(uint48).max);
|
|
161
|
-
|
|
162
|
-
vm.prank(_beneficiary);
|
|
163
|
-
_terminal.pay(_projectId, address(_usdc), usdcAmount, _beneficiary, 0, "", "");
|
|
164
|
-
|
|
165
|
-
// Check surplus (now includes all registered contexts automatically)
|
|
166
|
-
uint256 ethSurplus = _terminal.currentSurplusOf(_projectId, new address[](0), 18, _nativeCurrency);
|
|
167
|
-
|
|
168
|
-
// Total should include ETH amount + USDC converted to ETH
|
|
169
|
-
// 1 ETH + (2000 USDC / 2000 per ETH) = 2 ETH
|
|
170
|
-
// But the conversion depends on the feed direction
|
|
171
|
-
assertGt(ethSurplus, ethAmount, "total surplus should include both tokens");
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
/// @notice Balance tracking is per-token.
|
|
175
|
-
function test_perTokenBalance() public {
|
|
176
|
-
uint256 ethAmount = 3 ether;
|
|
177
|
-
uint256 usdcAmount = 1000e6;
|
|
178
|
-
|
|
179
|
-
// Pay ETH
|
|
180
|
-
vm.deal(_beneficiary, ethAmount);
|
|
181
|
-
vm.prank(_beneficiary);
|
|
182
|
-
_terminal.pay{value: ethAmount}(_projectId, JBConstants.NATIVE_TOKEN, ethAmount, _beneficiary, 0, "", "");
|
|
183
|
-
|
|
184
|
-
// Pay USDC
|
|
185
|
-
_usdc.mint(_beneficiary, usdcAmount);
|
|
186
|
-
vm.prank(_beneficiary);
|
|
187
|
-
_usdc.approve(address(permit2()), usdcAmount);
|
|
188
|
-
vm.prank(_beneficiary);
|
|
189
|
-
// forge-lint: disable-next-line(unsafe-typecast)
|
|
190
|
-
permit2().approve(address(_usdc), address(_terminal), uint160(usdcAmount), type(uint48).max);
|
|
191
|
-
|
|
192
|
-
vm.prank(_beneficiary);
|
|
193
|
-
_terminal.pay(_projectId, address(_usdc), usdcAmount, _beneficiary, 0, "", "");
|
|
194
|
-
|
|
195
|
-
// Check individual balances
|
|
196
|
-
uint256 ethBalance = jbTerminalStore().balanceOf(address(_terminal), _projectId, JBConstants.NATIVE_TOKEN);
|
|
197
|
-
assertEq(ethBalance, ethAmount, "ETH balance should match");
|
|
198
|
-
|
|
199
|
-
uint256 usdcBalance = jbTerminalStore().balanceOf(address(_terminal), _projectId, address(_usdc));
|
|
200
|
-
assertEq(usdcBalance, usdcAmount, "USDC balance should match");
|
|
201
|
-
}
|
|
202
|
-
}
|