@bananapus/core-v6 0.0.1
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/LICENSE +21 -0
- package/README.md +112 -0
- package/SKILLS.md +151 -0
- package/docs/book.css +13 -0
- package/docs/book.toml +12 -0
- package/docs/solidity.min.js +74 -0
- package/docs/src/README.md +703 -0
- package/docs/src/SUMMARY.md +94 -0
- package/docs/src/src/JBChainlinkV3PriceFeed.sol/contract.JBChainlinkV3PriceFeed.md +83 -0
- package/docs/src/src/JBChainlinkV3SequencerPriceFeed.sol/contract.JBChainlinkV3SequencerPriceFeed.md +88 -0
- package/docs/src/src/JBController.sol/contract.JBController.md +1121 -0
- package/docs/src/src/JBDeadline.sol/contract.JBDeadline.md +84 -0
- package/docs/src/src/JBDirectory.sol/contract.JBDirectory.md +294 -0
- package/docs/src/src/JBERC20.sol/contract.JBERC20.md +190 -0
- package/docs/src/src/JBFeelessAddresses.sol/contract.JBFeelessAddresses.md +80 -0
- package/docs/src/src/JBFundAccessLimits.sol/contract.JBFundAccessLimits.md +253 -0
- package/docs/src/src/JBMultiTerminal.sol/contract.JBMultiTerminal.md +1472 -0
- package/docs/src/src/JBPermissions.sol/contract.JBPermissions.md +199 -0
- package/docs/src/src/JBPrices.sol/contract.JBPrices.md +154 -0
- package/docs/src/src/JBProjects.sol/contract.JBProjects.md +131 -0
- package/docs/src/src/JBRulesets.sol/contract.JBRulesets.md +677 -0
- package/docs/src/src/JBSplits.sol/contract.JBSplits.md +237 -0
- package/docs/src/src/JBTerminalStore.sol/contract.JBTerminalStore.md +591 -0
- package/docs/src/src/JBTokens.sol/contract.JBTokens.md +353 -0
- package/docs/src/src/README.md +25 -0
- package/docs/src/src/abstract/JBControlled.sol/abstract.JBControlled.md +64 -0
- package/docs/src/src/abstract/JBPermissioned.sol/abstract.JBPermissioned.md +84 -0
- package/docs/src/src/abstract/README.md +5 -0
- package/docs/src/src/enums/JBApprovalStatus.sol/enum.JBApprovalStatus.md +17 -0
- package/docs/src/src/enums/README.md +4 -0
- package/docs/src/src/interfaces/IJBCashOutHook.sol/interface.IJBCashOutHook.md +29 -0
- package/docs/src/src/interfaces/IJBCashOutTerminal.sol/interface.IJBCashOutTerminal.md +57 -0
- package/docs/src/src/interfaces/IJBControlled.sol/interface.IJBControlled.md +12 -0
- package/docs/src/src/interfaces/IJBController.sol/interface.IJBController.md +334 -0
- package/docs/src/src/interfaces/IJBDirectory.sol/interface.IJBDirectory.md +108 -0
- package/docs/src/src/interfaces/IJBDirectoryAccessControl.sol/interface.IJBDirectoryAccessControl.md +19 -0
- package/docs/src/src/interfaces/IJBFeeTerminal.sol/interface.IJBFeeTerminal.md +91 -0
- package/docs/src/src/interfaces/IJBFeelessAddresses.sol/interface.IJBFeelessAddresses.md +26 -0
- package/docs/src/src/interfaces/IJBFundAccessLimits.sol/interface.IJBFundAccessLimits.md +88 -0
- package/docs/src/src/interfaces/IJBMigratable.sol/interface.IJBMigratable.md +29 -0
- package/docs/src/src/interfaces/IJBMultiTerminal.sol/interface.IJBMultiTerminal.md +50 -0
- package/docs/src/src/interfaces/IJBPayHook.sol/interface.IJBPayHook.md +28 -0
- package/docs/src/src/interfaces/IJBPayoutTerminal.sol/interface.IJBPayoutTerminal.md +105 -0
- package/docs/src/src/interfaces/IJBPermissioned.sol/interface.IJBPermissioned.md +12 -0
- package/docs/src/src/interfaces/IJBPermissions.sol/interface.IJBPermissions.md +74 -0
- package/docs/src/src/interfaces/IJBPermitTerminal.sol/interface.IJBPermitTerminal.md +15 -0
- package/docs/src/src/interfaces/IJBPriceFeed.sol/interface.IJBPriceFeed.md +12 -0
- package/docs/src/src/interfaces/IJBPrices.sol/interface.IJBPrices.md +74 -0
- package/docs/src/src/interfaces/IJBProjectUriRegistry.sol/interface.IJBProjectUriRegistry.md +19 -0
- package/docs/src/src/interfaces/IJBProjects.sol/interface.IJBProjects.md +49 -0
- package/docs/src/src/interfaces/IJBRulesetApprovalHook.sol/interface.IJBRulesetApprovalHook.md +35 -0
- package/docs/src/src/interfaces/IJBRulesetDataHook.sol/interface.IJBRulesetDataHook.md +97 -0
- package/docs/src/src/interfaces/IJBRulesets.sol/interface.IJBRulesets.md +165 -0
- package/docs/src/src/interfaces/IJBSplitHook.sol/interface.IJBSplitHook.md +31 -0
- package/docs/src/src/interfaces/IJBSplits.sol/interface.IJBSplits.md +35 -0
- package/docs/src/src/interfaces/IJBTerminal.sol/interface.IJBTerminal.md +141 -0
- package/docs/src/src/interfaces/IJBTerminalStore.sol/interface.IJBTerminalStore.md +198 -0
- package/docs/src/src/interfaces/IJBToken.sol/interface.IJBToken.md +54 -0
- package/docs/src/src/interfaces/IJBTokenUriResolver.sol/interface.IJBTokenUriResolver.md +12 -0
- package/docs/src/src/interfaces/IJBTokens.sol/interface.IJBTokens.md +151 -0
- package/docs/src/src/interfaces/README.md +33 -0
- package/docs/src/src/libraries/JBCashOuts.sol/library.JBCashOuts.md +40 -0
- package/docs/src/src/libraries/JBConstants.sol/library.JBConstants.md +52 -0
- package/docs/src/src/libraries/JBCurrencyIds.sol/library.JBCurrencyIds.md +19 -0
- package/docs/src/src/libraries/JBFees.sol/library.JBFees.md +52 -0
- package/docs/src/src/libraries/JBFixedPointNumber.sol/library.JBFixedPointNumber.md +12 -0
- package/docs/src/src/libraries/JBMetadataResolver.sol/library.JBMetadataResolver.md +242 -0
- package/docs/src/src/libraries/JBRulesetMetadataResolver.sol/library.JBRulesetMetadataResolver.md +180 -0
- package/docs/src/src/libraries/JBSplitGroupIds.sol/library.JBSplitGroupIds.md +14 -0
- package/docs/src/src/libraries/JBSurplus.sol/library.JBSurplus.md +44 -0
- package/docs/src/src/libraries/README.md +12 -0
- package/docs/src/src/periphery/JBDeadline1Day.sol/contract.JBDeadline1Day.md +15 -0
- package/docs/src/src/periphery/JBDeadline3Days.sol/contract.JBDeadline3Days.md +15 -0
- package/docs/src/src/periphery/JBDeadline3Hours.sol/contract.JBDeadline3Hours.md +15 -0
- package/docs/src/src/periphery/JBDeadline7Days.sol/contract.JBDeadline7Days.md +15 -0
- package/docs/src/src/periphery/JBMatchingPriceFeed.sol/contract.JBMatchingPriceFeed.md +22 -0
- package/docs/src/src/periphery/README.md +8 -0
- package/docs/src/src/structs/JBAccountingContext.sol/struct.JBAccountingContext.md +20 -0
- package/docs/src/src/structs/JBAfterCashOutRecordedContext.sol/struct.JBAfterCashOutRecordedContext.md +43 -0
- package/docs/src/src/structs/JBAfterPayRecordedContext.sol/struct.JBAfterPayRecordedContext.md +42 -0
- package/docs/src/src/structs/JBBeforeCashOutRecordedContext.sol/struct.JBBeforeCashOutRecordedContext.md +45 -0
- package/docs/src/src/structs/JBBeforePayRecordedContext.sol/struct.JBBeforePayRecordedContext.md +41 -0
- package/docs/src/src/structs/JBCashOutHookSpecification.sol/struct.JBCashOutHookSpecification.md +22 -0
- package/docs/src/src/structs/JBCurrencyAmount.sol/struct.JBCurrencyAmount.md +17 -0
- package/docs/src/src/structs/JBFee.sol/struct.JBFee.md +20 -0
- package/docs/src/src/structs/JBFundAccessLimitGroup.sol/struct.JBFundAccessLimitGroup.md +39 -0
- package/docs/src/src/structs/JBPayHookSpecification.sol/struct.JBPayHookSpecification.md +22 -0
- package/docs/src/src/structs/JBPermissionsData.sol/struct.JBPermissionsData.md +21 -0
- package/docs/src/src/structs/JBRuleset.sol/struct.JBRuleset.md +55 -0
- package/docs/src/src/structs/JBRulesetConfig.sol/struct.JBRulesetConfig.md +51 -0
- package/docs/src/src/structs/JBRulesetMetadata.sol/struct.JBRulesetMetadata.md +79 -0
- package/docs/src/src/structs/JBRulesetWeightCache.sol/struct.JBRulesetWeightCache.md +16 -0
- package/docs/src/src/structs/JBRulesetWithMetadata.sol/struct.JBRulesetWithMetadata.md +16 -0
- package/docs/src/src/structs/JBSingleAllowance.sol/struct.JBSingleAllowance.md +26 -0
- package/docs/src/src/structs/JBSplit.sol/struct.JBSplit.md +49 -0
- package/docs/src/src/structs/JBSplitGroup.sol/struct.JBSplitGroup.md +17 -0
- package/docs/src/src/structs/JBSplitHookContext.sol/struct.JBSplitHookContext.md +29 -0
- package/docs/src/src/structs/JBTerminalConfig.sol/struct.JBTerminalConfig.md +16 -0
- package/docs/src/src/structs/JBTokenAmount.sol/struct.JBTokenAmount.md +23 -0
- package/docs/src/src/structs/README.md +25 -0
- package/foundry.lock +11 -0
- package/foundry.toml +41 -0
- package/package.json +38 -0
- package/remappings.txt +1 -0
- package/script/Deploy.s.sol +111 -0
- package/script/DeployPeriphery.s.sol +287 -0
- package/script/helpers/CoreDeploymentLib.sol +121 -0
- package/slither-ci.config.json +10 -0
- package/sphinx.lock +507 -0
- package/src/JBChainlinkV3PriceFeed.sol +77 -0
- package/src/JBChainlinkV3SequencerPriceFeed.sol +75 -0
- package/src/JBController.sol +1186 -0
- package/src/JBDeadline.sol +73 -0
- package/src/JBDirectory.sol +343 -0
- package/src/JBERC20.sol +131 -0
- package/src/JBFeelessAddresses.sol +54 -0
- package/src/JBFundAccessLimits.sol +308 -0
- package/src/JBMultiTerminal.sol +2024 -0
- package/src/JBPermissions.sol +252 -0
- package/src/JBPrices.sol +227 -0
- package/src/JBProjects.sol +126 -0
- package/src/JBRulesets.sol +1093 -0
- package/src/JBSplits.sol +324 -0
- package/src/JBTerminalStore.sol +908 -0
- package/src/JBTokens.sol +376 -0
- package/src/abstract/JBControlled.sol +48 -0
- package/src/abstract/JBPermissioned.sol +77 -0
- package/src/enums/JBApprovalStatus.sol +12 -0
- package/src/interfaces/IJBCashOutHook.sol +15 -0
- package/src/interfaces/IJBCashOutTerminal.sol +51 -0
- package/src/interfaces/IJBControlled.sol +10 -0
- package/src/interfaces/IJBController.sol +280 -0
- package/src/interfaces/IJBDirectory.sol +69 -0
- package/src/interfaces/IJBDirectoryAccessControl.sol +15 -0
- package/src/interfaces/IJBFeeTerminal.sol +61 -0
- package/src/interfaces/IJBFeelessAddresses.sol +17 -0
- package/src/interfaces/IJBFundAccessLimits.sol +94 -0
- package/src/interfaces/IJBMigratable.sol +24 -0
- package/src/interfaces/IJBMultiTerminal.sol +36 -0
- package/src/interfaces/IJBPayHook.sol +14 -0
- package/src/interfaces/IJBPayoutTerminal.sol +92 -0
- package/src/interfaces/IJBPermissioned.sol +10 -0
- package/src/interfaces/IJBPermissions.sol +71 -0
- package/src/interfaces/IJBPermitTerminal.sol +14 -0
- package/src/interfaces/IJBPriceFeed.sol +10 -0
- package/src/interfaces/IJBPrices.sol +65 -0
- package/src/interfaces/IJBProjectUriRegistry.sol +15 -0
- package/src/interfaces/IJBProjects.sol +27 -0
- package/src/interfaces/IJBRulesetApprovalHook.sol +21 -0
- package/src/interfaces/IJBRulesetDataHook.sol +56 -0
- package/src/interfaces/IJBRulesets.sol +151 -0
- package/src/interfaces/IJBSplitHook.sol +16 -0
- package/src/interfaces/IJBSplits.sol +28 -0
- package/src/interfaces/IJBTerminal.sol +120 -0
- package/src/interfaces/IJBTerminalStore.sol +225 -0
- package/src/interfaces/IJBToken.sol +39 -0
- package/src/interfaces/IJBTokenUriResolver.sol +10 -0
- package/src/interfaces/IJBTokens.sol +113 -0
- package/src/libraries/JBCashOuts.sol +120 -0
- package/src/libraries/JBConstants.sol +14 -0
- package/src/libraries/JBCurrencyIds.sol +7 -0
- package/src/libraries/JBFees.sol +28 -0
- package/src/libraries/JBFixedPointNumber.sol +12 -0
- package/src/libraries/JBMetadataResolver.sol +306 -0
- package/src/libraries/JBRulesetMetadataResolver.sol +160 -0
- package/src/libraries/JBSplitGroupIds.sol +7 -0
- package/src/libraries/JBSurplus.sol +40 -0
- package/src/periphery/JBDeadline1Day.sol +8 -0
- package/src/periphery/JBDeadline3Days.sol +8 -0
- package/src/periphery/JBDeadline3Hours.sol +8 -0
- package/src/periphery/JBDeadline7Days.sol +8 -0
- package/src/periphery/JBMatchingPriceFeed.sol +13 -0
- package/src/structs/JBAccountingContext.sol +12 -0
- package/src/structs/JBAfterCashOutRecordedContext.sol +30 -0
- package/src/structs/JBAfterPayRecordedContext.sol +29 -0
- package/src/structs/JBBeforeCashOutRecordedContext.sol +31 -0
- package/src/structs/JBBeforePayRecordedContext.sol +28 -0
- package/src/structs/JBCashOutHookSpecification.sol +15 -0
- package/src/structs/JBCurrencyAmount.sol +10 -0
- package/src/structs/JBFee.sol +12 -0
- package/src/structs/JBFundAccessLimitGroup.sol +28 -0
- package/src/structs/JBPayHookSpecification.sol +15 -0
- package/src/structs/JBPermissionsData.sol +13 -0
- package/src/structs/JBRuleset.sol +42 -0
- package/src/structs/JBRulesetConfig.sol +43 -0
- package/src/structs/JBRulesetMetadata.sol +56 -0
- package/src/structs/JBRulesetWeightCache.sol +9 -0
- package/src/structs/JBRulesetWithMetadata.sol +12 -0
- package/src/structs/JBSingleAllowance.sol +16 -0
- package/src/structs/JBSplit.sol +37 -0
- package/src/structs/JBSplitGroup.sol +12 -0
- package/src/structs/JBSplitHookContext.sol +20 -0
- package/src/structs/JBTerminalConfig.sol +12 -0
- package/src/structs/JBTokenAmount.sol +14 -0
- package/test/AuditExploits.t.sol +2710 -0
- package/test/ComprehensiveInvariant.t.sol +298 -0
- package/test/EconomicSimulation.t.sol +340 -0
- package/test/EntryPointPermutations.t.sol +671 -0
- package/test/FlashLoanAttacks.t.sol +792 -0
- package/test/PermissionEscalation.t.sol +679 -0
- package/test/RulesetTransitions.t.sol +699 -0
- package/test/SplitLoopTests.t.sol +731 -0
- package/test/TestAccessToFunds.sol +2644 -0
- package/test/TestCashOut.sol +185 -0
- package/test/TestCashOutCountFor.sol +272 -0
- package/test/TestCashOutHooks.sol +317 -0
- package/test/TestCashOutTimingEdge.sol +229 -0
- package/test/TestDurationUnderflow.sol +220 -0
- package/test/TestFeeProcessingFailure.sol +208 -0
- package/test/TestFees.sol +604 -0
- package/test/TestInterfaceSupport.sol +62 -0
- package/test/TestJBERC20Inheritance.sol +91 -0
- package/test/TestLaunchProject.sol +176 -0
- package/test/TestMetaTx.sol +203 -0
- package/test/TestMetadataParserLib.sol +438 -0
- package/test/TestMigrationHeldFees.sol +249 -0
- package/test/TestMintTokensOf.sol +172 -0
- package/test/TestMultiTokenSurplus.sol +206 -0
- package/test/TestMultipleAccessLimits.sol +642 -0
- package/test/TestPayBurnRedeemFlow.sol +180 -0
- package/test/TestPayHooks.sol +190 -0
- package/test/TestPermissions.sol +305 -0
- package/test/TestPermissionsEdge.sol +286 -0
- package/test/TestPermit2Terminal.sol +339 -0
- package/test/TestRulesetQueueing.sol +1001 -0
- package/test/TestRulesetQueuingStress.sol +778 -0
- package/test/TestRulesetWeightCaching.sol +177 -0
- package/test/TestSplits.sol +369 -0
- package/test/TestTerminalMigration.sol +167 -0
- package/test/TestTokenFlow.sol +174 -0
- package/test/WeirdTokenTests.t.sol +764 -0
- package/test/formal/BondingCurveProperties.t.sol +411 -0
- package/test/formal/FeeProperties.t.sol +246 -0
- package/test/helpers/JBTest.sol +129 -0
- package/test/helpers/MetadataResolverHelper.sol +116 -0
- package/test/helpers/TestBaseWorkflow.sol +317 -0
- package/test/invariants/Phase3DeepInvariant.t.sol +404 -0
- package/test/invariants/RulesetsInvariant.t.sol +115 -0
- package/test/invariants/TerminalStoreInvariant.t.sol +220 -0
- package/test/invariants/TokensInvariant.t.sol +184 -0
- package/test/invariants/handlers/ComprehensiveHandler.sol +285 -0
- package/test/invariants/handlers/EconomicHandler.sol +347 -0
- package/test/invariants/handlers/Phase3Handler.sol +414 -0
- package/test/invariants/handlers/RulesetsHandler.sol +111 -0
- package/test/invariants/handlers/TerminalStoreHandler.sol +146 -0
- package/test/invariants/handlers/TokensHandler.sol +127 -0
- package/test/mock/ERC2771ForwarderMock.sol +37 -0
- package/test/mock/MockERC20.sol +18 -0
- package/test/mock/MockMaliciousBeneficiary.sol +67 -0
- package/test/mock/MockMaliciousSplitHook.sol +42 -0
- package/test/mock/MockPriceFeed.sol +20 -0
- package/test/trees/JBController/burnTokensOf.tree +9 -0
- package/test/trees/JBController/claimTokensFor.tree +5 -0
- package/test/trees/JBController/deployERC20For.tree +5 -0
- package/test/trees/JBController/getRulesetOf.tree +5 -0
- package/test/trees/JBController/launchProjectFor.tree +12 -0
- package/test/trees/JBController/launchRulesetsFor.tree +8 -0
- package/test/trees/JBController/migrateController.tree +12 -0
- package/test/trees/JBController/mintTokensOf.tree +12 -0
- package/test/trees/JBController/payReservedTokenToTerminal.tree +8 -0
- package/test/trees/JBController/receiveMigrationFrom.tree +4 -0
- package/test/trees/JBController/sendReservedTokensToSplitsOf.tree +12 -0
- package/test/trees/JBController/setMetadataOf.tree +5 -0
- package/test/trees/JBController/setSplitGroupsOf.tree +5 -0
- package/test/trees/JBController/setTokenFor.tree +5 -0
- package/test/trees/JBController/transferCreditsFrom.tree +8 -0
- package/test/trees/JBDirectory/primaryTerminalOf.tree +8 -0
- package/test/trees/JBDirectory/setControllerOf.tree +11 -0
- package/test/trees/JBDirectory/setPrimaryTerminalOf.tree +15 -0
- package/test/trees/JBDirectory/setTerminalsOf.tree +11 -0
- package/test/trees/JBERC20/initialize.tree +7 -0
- package/test/trees/JBERC20/name.tree +5 -0
- package/test/trees/JBERC20/nonces.tree +5 -0
- package/test/trees/JBERC20/symbol.tree +5 -0
- package/test/trees/JBFeelessAddresses/setFeelessAddress.tree +5 -0
- package/test/trees/JBFeelessAddresses/supportsInterface.tree +5 -0
- package/test/trees/JBFundAccessLimits/payoutLimitOf.tree +5 -0
- package/test/trees/JBFundAccessLimits/payoutLimitsOf.tree +8 -0
- package/test/trees/JBFundAccessLimits/setFundAccessLimitsFor.tree +18 -0
- package/test/trees/JBFundAccessLimits/surplusAllowanceOf.tree +5 -0
- package/test/trees/JBFundAccessLimits/surplusAllowancesOf.tree +8 -0
- package/test/trees/JBMetadataResolver/getDataFor.tree +8 -0
- package/test/trees/JBMultiTerminal/accountingContextsOf.tree +5 -0
- package/test/trees/JBMultiTerminal/addAccountingContextsFor.tree +10 -0
- package/test/trees/JBMultiTerminal/addToBalanceOf.tree +23 -0
- package/test/trees/JBMultiTerminal/cashOutTokensOf.tree +23 -0
- package/test/trees/JBMultiTerminal/executePayout.tree +32 -0
- package/test/trees/JBMultiTerminal/executeProcessFee.tree +14 -0
- package/test/trees/JBMultiTerminal/migrateBalanceOf.tree +12 -0
- package/test/trees/JBMultiTerminal/pay.tree +23 -0
- package/test/trees/JBMultiTerminal/processHeldFeesOf.tree +8 -0
- package/test/trees/JBMultiTerminal/sendPayoutsOf.tree +34 -0
- package/test/trees/JBMultiTerminal/useAllowanceOf.tree +16 -0
- package/test/trees/JBPermissions/hasPermission.tree +8 -0
- package/test/trees/JBPermissions/hasPermissions.tree +8 -0
- package/test/trees/JBPermissions/setPermissionsFor.tree +5 -0
- package/test/trees/JBPrices/addPriceFeedFor.tree +14 -0
- package/test/trees/JBPrices/pricePerUnitOf.tree +11 -0
- package/test/trees/JBProjects/createFor.tree +11 -0
- package/test/trees/JBProjects/setTokenUriResolver.tree +5 -0
- package/test/trees/JBProjects/supportsInterface.tree +9 -0
- package/test/trees/JBProjects/tokenURI.tree +5 -0
- package/test/trees/JBRulesets/currentApprovalStatusForLatestRulesetOf.tree +8 -0
- package/test/trees/JBRulesets/currentOf.tree +12 -0
- package/test/trees/JBRulesets/getRulesetOf.tree +5 -0
- package/test/trees/JBRulesets/latestQueuedRulesetOf.tree +10 -0
- package/test/trees/JBRulesets/rulesetsOf.tree +11 -0
- package/test/trees/JBRulesets/upcomingRulesetOf.tree +20 -0
- package/test/trees/JBRulesets/updateRulesetWeightCache.tree +5 -0
- package/test/trees/JBSplits/setSplitGroupsOf.tree +17 -0
- package/test/trees/JBSplits/splitsOf.tree +5 -0
- package/test/trees/JBTerminalStore/currentReclaimableSurplusOf.tree +16 -0
- package/test/trees/JBTerminalStore/currentSurplusOf.tree +25 -0
- package/test/trees/JBTerminalStore/currentTotalSurplusOf.tree +5 -0
- package/test/trees/JBTerminalStore/recordCashOutsFor.tree +16 -0
- package/test/trees/JBTerminalStore/recordPaymentFrom.tree +14 -0
- package/test/trees/JBTerminalStore/recordPayoutFor.tree +10 -0
- package/test/trees/JBTerminalStore/recordTerminalMigration.tree +5 -0
- package/test/trees/JBTerminalStore/recordUsedAllowanceOf.tree +10 -0
- package/test/trees/JBTokens/burnFrom.tree +10 -0
- package/test/trees/JBTokens/claimTokensFor.tree +10 -0
- package/test/trees/JBTokens/deployERC20For.tree +12 -0
- package/test/trees/JBTokens/mintFor.tree +10 -0
- package/test/trees/JBTokens/setTokenFor.tree +11 -0
- package/test/trees/JBTokens/totalBalanceOf.tree +5 -0
- package/test/trees/JBTokens/totalSupplyOf.tree +5 -0
- package/test/trees/JBTokens/transferCreditsFrom.tree +8 -0
- package/test/trees/mintTokensOf.tree +12 -0
- package/test/units/static/JBChainlinkV3PriceFeed/TestPriceFeed.sol +220 -0
- package/test/units/static/JBController/JBControllerSetup.sol +40 -0
- package/test/units/static/JBController/TestBurnTokensOf.sol +107 -0
- package/test/units/static/JBController/TestClaimTokensFor.sol +60 -0
- package/test/units/static/JBController/TestDeployErc20For.sol +80 -0
- package/test/units/static/JBController/TestLaunchProjectFor.sol +282 -0
- package/test/units/static/JBController/TestLaunchRulesetsFor.sol +322 -0
- package/test/units/static/JBController/TestMigrateController.sol +148 -0
- package/test/units/static/JBController/TestMintTokensOfUnits.sol +102 -0
- package/test/units/static/JBController/TestPayReservedTokenToTerminal.sol +71 -0
- package/test/units/static/JBController/TestReceiveMigrationFrom.sol +95 -0
- package/test/units/static/JBController/TestRulesetViews.sol +219 -0
- package/test/units/static/JBController/TestSendReservedTokensToSplitsOf.sol +595 -0
- package/test/units/static/JBController/TestSetSplitGroupsOf.sol +63 -0
- package/test/units/static/JBController/TestSetTokenFor.sol +227 -0
- package/test/units/static/JBController/TestSetUriOf.sol +53 -0
- package/test/units/static/JBController/TestTransferCreditsFrom.sol +159 -0
- package/test/units/static/JBDeadline/TestDeadlineFuzz.sol +194 -0
- package/test/units/static/JBDirectory/JBDirectorySetup.sol +22 -0
- package/test/units/static/JBDirectory/TestPrimaryTerminalOf.sol +122 -0
- package/test/units/static/JBDirectory/TestSetControllerOf.sol +173 -0
- package/test/units/static/JBDirectory/TestSetControllerOfMigrationOrder.sol +98 -0
- package/test/units/static/JBDirectory/TestSetPrimaryTerminalOf.sol +169 -0
- package/test/units/static/JBDirectory/TestSetTerminalsOf.sol +128 -0
- package/test/units/static/JBERC20/JBERC20Setup.sol +20 -0
- package/test/units/static/JBERC20/SigUtils.sol +34 -0
- package/test/units/static/JBERC20/TestInitialize.sol +54 -0
- package/test/units/static/JBERC20/TestName.sol +30 -0
- package/test/units/static/JBERC20/TestNonces.sol +59 -0
- package/test/units/static/JBERC20/TestSymbol.sol +31 -0
- package/test/units/static/JBFeelessAdresses/JBFeelessSetup.sol +20 -0
- package/test/units/static/JBFeelessAdresses/TestInterfaces.sol +29 -0
- package/test/units/static/JBFeelessAdresses/TestSetFeelessAddress.sol +35 -0
- package/test/units/static/JBFees/TestFeesFuzz.sol +78 -0
- package/test/units/static/JBFixedPointNumber/TestAdjustDecimals.sol +16 -0
- package/test/units/static/JBFixedPointNumber/TestAdjustDecimalsFuzz.sol +71 -0
- package/test/units/static/JBFundAccessLimits/JBFundAccessSetup.sol +21 -0
- package/test/units/static/JBFundAccessLimits/TestFundAccessLimitsEdge.sol +159 -0
- package/test/units/static/JBFundAccessLimits/TestPayoutLimitOf.sol +56 -0
- package/test/units/static/JBFundAccessLimits/TestPayoutLimitsOf.sol +94 -0
- package/test/units/static/JBFundAccessLimits/TestSetFundAccessLimitsFor.sol +182 -0
- package/test/units/static/JBFundAccessLimits/TestSurplusAllowanceOf.sol +61 -0
- package/test/units/static/JBFundAccessLimits/TestSurplusAllowancesOf.sol +96 -0
- package/test/units/static/JBMetadataResolver/TestGetDataFor.sol +89 -0
- package/test/units/static/JBMetadataResolver/TestMetadataResolverFuzz.sol +227 -0
- package/test/units/static/JBMetadataResolver/TestMetadataResolverM20M21.sol +245 -0
- package/test/units/static/JBMultiTerminal/JBMultiTerminalSetup.sol +39 -0
- package/test/units/static/JBMultiTerminal/TestAccountingContextsOf.sol +65 -0
- package/test/units/static/JBMultiTerminal/TestAddAccountingContextsFor.sol +313 -0
- package/test/units/static/JBMultiTerminal/TestAddToBalanceOf.sol +432 -0
- package/test/units/static/JBMultiTerminal/TestCashOutTokensOf.sol +478 -0
- package/test/units/static/JBMultiTerminal/TestExecutePayout.sol +577 -0
- package/test/units/static/JBMultiTerminal/TestExecuteProcessFee.sol +176 -0
- package/test/units/static/JBMultiTerminal/TestMigrateBalanceOf.sol +190 -0
- package/test/units/static/JBMultiTerminal/TestPay.sol +514 -0
- package/test/units/static/JBMultiTerminal/TestProcessHeldFeesOf.sol +29 -0
- package/test/units/static/JBMultiTerminal/TestSendPayoutsOf.sol +243 -0
- package/test/units/static/JBMultiTerminal/TestUseAllowanceOf.sol +310 -0
- package/test/units/static/JBPermissions/JBPermissionsSetup.sol +18 -0
- package/test/units/static/JBPermissions/TestHasPermission.sol +50 -0
- package/test/units/static/JBPermissions/TestHasPermissions.sol +93 -0
- package/test/units/static/JBPermissions/TestSetPermissionsFor.sol +62 -0
- package/test/units/static/JBPrices/JBPricesSetup.sol +26 -0
- package/test/units/static/JBPrices/TestAddPriceFeedFor.sol +102 -0
- package/test/units/static/JBPrices/TestPricePerUnitOf.sol +129 -0
- package/test/units/static/JBPrices/TestPrices.sol +262 -0
- package/test/units/static/JBProjects/JBProjectsSetup.sol +20 -0
- package/test/units/static/JBProjects/TestCreateFor.sol +69 -0
- package/test/units/static/JBProjects/TestInitialProject.sol +19 -0
- package/test/units/static/JBProjects/TestInterfaces.sol +27 -0
- package/test/units/static/JBProjects/TestSetResolver.sol +36 -0
- package/test/units/static/JBProjects/TestTokenUri.sol +38 -0
- package/test/units/static/JBRulesetMetadataResolver/TestSetCashOutTaxRateTo.sol +99 -0
- package/test/units/static/JBRulesets/JBRulesetsSetup.sol +21 -0
- package/test/units/static/JBRulesets/TestCurrentApprovalStatusForLatestRulesetOf.sol +257 -0
- package/test/units/static/JBRulesets/TestCurrentOf.sol +231 -0
- package/test/units/static/JBRulesets/TestGetRulesetOf.sol +94 -0
- package/test/units/static/JBRulesets/TestLatestQueuedRulesetOf.sol +252 -0
- package/test/units/static/JBRulesets/TestRulesets.sol +617 -0
- package/test/units/static/JBRulesets/TestRulesetsOf.sol +37 -0
- package/test/units/static/JBRulesets/TestUpcomingRulesetOf.sol +526 -0
- package/test/units/static/JBRulesets/TestUpdateRulesetWeightCache.sol +91 -0
- package/test/units/static/JBSplits/JBSplitsSetup.sol +23 -0
- package/test/units/static/JBSplits/TestSelfManagedSplitGroups.sol +502 -0
- package/test/units/static/JBSplits/TestSetSplitGroupsOf.sol +370 -0
- package/test/units/static/JBSplits/TestSplitsLockedEdge.sol +262 -0
- package/test/units/static/JBSplits/TestSplitsOf.sol +24 -0
- package/test/units/static/JBSplits/TestSplitsPacking.sol +33 -0
- package/test/units/static/JBSurplus/TestSurplusFuzz.sol +125 -0
- package/test/units/static/JBTerminalStore/JBTerminalStoreSetup.sol +23 -0
- package/test/units/static/JBTerminalStore/TestCurrentReclaimableSurplusOf.sol +434 -0
- package/test/units/static/JBTerminalStore/TestCurrentSurplusOf.sol +428 -0
- package/test/units/static/JBTerminalStore/TestCurrentTotalSurplusOf.sol +65 -0
- package/test/units/static/JBTerminalStore/TestRecordCashOutsFor.sol +479 -0
- package/test/units/static/JBTerminalStore/TestRecordPaymentFrom.sol +508 -0
- package/test/units/static/JBTerminalStore/TestRecordPayoutFor.sol +257 -0
- package/test/units/static/JBTerminalStore/TestRecordTerminalMigration.sol +131 -0
- package/test/units/static/JBTerminalStore/TestRecordUsedAllowanceOf.sol +390 -0
- package/test/units/static/JBTerminalStore/TestUint224Overflow.sol +187 -0
- package/test/units/static/JBTokens/JBTokensSetup.sol +23 -0
- package/test/units/static/JBTokens/TestBurnFrom.sol +104 -0
- package/test/units/static/JBTokens/TestClaimTokensFor.sol +107 -0
- package/test/units/static/JBTokens/TestDeployERC20ForUnits.sol +89 -0
- package/test/units/static/JBTokens/TestMintFor.sol +97 -0
- package/test/units/static/JBTokens/TestSetTokenFor.sol +95 -0
- package/test/units/static/JBTokens/TestTotalBalanceOf.sol +65 -0
- package/test/units/static/JBTokens/TestTotalSupplyOf.sol +56 -0
- package/test/units/static/JBTokens/TestTransferCreditsFrom.sol +54 -0
|
@@ -0,0 +1,414 @@
|
|
|
1
|
+
// SPDX-License-Identifier: MIT
|
|
2
|
+
pragma solidity ^0.8.6;
|
|
3
|
+
|
|
4
|
+
import "forge-std/Test.sol";
|
|
5
|
+
import {JBConstants} from "../../../src/libraries/JBConstants.sol";
|
|
6
|
+
import {JBFees} from "../../../src/libraries/JBFees.sol";
|
|
7
|
+
import {JBFee} from "../../../src/structs/JBFee.sol";
|
|
8
|
+
import {IJBMultiTerminal} from "../../../src/interfaces/IJBMultiTerminal.sol";
|
|
9
|
+
import {IJBTerminalStore} from "../../../src/interfaces/IJBTerminalStore.sol";
|
|
10
|
+
import {IJBController} from "../../../src/interfaces/IJBController.sol";
|
|
11
|
+
import {IJBTokens} from "../../../src/interfaces/IJBTokens.sol";
|
|
12
|
+
import {JBMultiTerminal} from "../../../src/JBMultiTerminal.sol";
|
|
13
|
+
|
|
14
|
+
/// @title Phase3Handler
|
|
15
|
+
/// @notice Stateful fuzzing handler with ghost variable tracking for strict invariant verification.
|
|
16
|
+
/// @dev 14 operations across 4 projects with 10 actors. Ghost variables track every fee flow,
|
|
17
|
+
/// enabling strict equality checks instead of the looser `>=` used in Phase 2.
|
|
18
|
+
contract Phase3Handler is Test {
|
|
19
|
+
IJBMultiTerminal public terminal;
|
|
20
|
+
IJBTerminalStore public store;
|
|
21
|
+
IJBController public controller;
|
|
22
|
+
IJBTokens public tokens;
|
|
23
|
+
|
|
24
|
+
uint256 public constant PROJECT_FEE = 1;
|
|
25
|
+
uint256 public projectId2;
|
|
26
|
+
uint256 public projectId3;
|
|
27
|
+
uint256 public projectId4;
|
|
28
|
+
address public projectOwner;
|
|
29
|
+
|
|
30
|
+
uint256 public constant FEE_PERCENT = 25; // 2.5% — matches JBMultiTerminal.FEE
|
|
31
|
+
|
|
32
|
+
// =========================================================================
|
|
33
|
+
// Actors
|
|
34
|
+
// =========================================================================
|
|
35
|
+
address[] public actors;
|
|
36
|
+
uint256 public constant NUM_ACTORS = 10;
|
|
37
|
+
|
|
38
|
+
// =========================================================================
|
|
39
|
+
// Ghost Variables — strict accounting
|
|
40
|
+
// =========================================================================
|
|
41
|
+
|
|
42
|
+
// Per-project total inflows/outflows
|
|
43
|
+
mapping(uint256 => uint256) public ghost_totalPaidIn;
|
|
44
|
+
mapping(uint256 => uint256) public ghost_totalCashedOut;
|
|
45
|
+
mapping(uint256 => uint256) public ghost_totalPaidOut;
|
|
46
|
+
mapping(uint256 => uint256) public ghost_totalAllowanceUsed;
|
|
47
|
+
mapping(uint256 => uint256) public ghost_totalAddedToBalance;
|
|
48
|
+
|
|
49
|
+
// Fee tracking (key innovation for strict invariants)
|
|
50
|
+
mapping(uint256 => uint256) public ghost_totalFeesDeducted;
|
|
51
|
+
mapping(uint256 => uint256) public ghost_totalFeesSentToProject1;
|
|
52
|
+
mapping(uint256 => uint256) public ghost_totalHeldFeeAmounts;
|
|
53
|
+
mapping(uint256 => uint256) public ghost_totalReturnedFees;
|
|
54
|
+
mapping(uint256 => uint256) public ghost_totalProcessedFees;
|
|
55
|
+
|
|
56
|
+
// Per-actor tracking
|
|
57
|
+
mapping(address => mapping(uint256 => uint256)) public ghost_actorContributed;
|
|
58
|
+
mapping(address => mapping(uint256 => uint256)) public ghost_actorExtracted;
|
|
59
|
+
|
|
60
|
+
// Token tracking
|
|
61
|
+
mapping(uint256 => uint256) public ghost_totalReservesSent;
|
|
62
|
+
|
|
63
|
+
// Global conservation
|
|
64
|
+
uint256 public ghost_globalInflows;
|
|
65
|
+
uint256 public ghost_globalOutflows;
|
|
66
|
+
|
|
67
|
+
// Operation counters
|
|
68
|
+
uint256 public callCount_pay2;
|
|
69
|
+
uint256 public callCount_pay3;
|
|
70
|
+
uint256 public callCount_cashOut2;
|
|
71
|
+
uint256 public callCount_cashOut3;
|
|
72
|
+
uint256 public callCount_sendPayouts2;
|
|
73
|
+
uint256 public callCount_useAllowance2;
|
|
74
|
+
uint256 public callCount_sendReserved2;
|
|
75
|
+
uint256 public callCount_processHeldFees2;
|
|
76
|
+
uint256 public callCount_addToBalanceReturn2;
|
|
77
|
+
uint256 public callCount_addToBalanceNoReturn2;
|
|
78
|
+
uint256 public callCount_burnTokens2;
|
|
79
|
+
uint256 public callCount_burnTokens3;
|
|
80
|
+
uint256 public callCount_claimCredits2;
|
|
81
|
+
uint256 public callCount_advanceTime;
|
|
82
|
+
|
|
83
|
+
constructor(
|
|
84
|
+
IJBMultiTerminal _terminal,
|
|
85
|
+
IJBTerminalStore _store,
|
|
86
|
+
IJBController _controller,
|
|
87
|
+
IJBTokens _tokens,
|
|
88
|
+
uint256 _projectId2,
|
|
89
|
+
uint256 _projectId3,
|
|
90
|
+
uint256 _projectId4,
|
|
91
|
+
address _projectOwner
|
|
92
|
+
) {
|
|
93
|
+
terminal = _terminal;
|
|
94
|
+
store = _store;
|
|
95
|
+
controller = _controller;
|
|
96
|
+
tokens = _tokens;
|
|
97
|
+
projectId2 = _projectId2;
|
|
98
|
+
projectId3 = _projectId3;
|
|
99
|
+
projectId4 = _projectId4;
|
|
100
|
+
projectOwner = _projectOwner;
|
|
101
|
+
|
|
102
|
+
for (uint256 i = 0; i < NUM_ACTORS; i++) {
|
|
103
|
+
address actor = address(uint160(0x4000 + i));
|
|
104
|
+
actors.push(actor);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
// =========================================================================
|
|
109
|
+
// Helpers
|
|
110
|
+
// =========================================================================
|
|
111
|
+
|
|
112
|
+
function _getActor(uint256 seed) internal view returns (address) {
|
|
113
|
+
return actors[seed % actors.length];
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
/// @notice Compute the fee that would be deducted from `amount` (forward fee).
|
|
117
|
+
function _feeFrom(uint256 amount) internal pure returns (uint256) {
|
|
118
|
+
return JBFees.feeAmountFrom(amount, FEE_PERCENT);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// =========================================================================
|
|
122
|
+
// 14 Operations
|
|
123
|
+
// =========================================================================
|
|
124
|
+
|
|
125
|
+
/// @notice Pay into project 2.
|
|
126
|
+
function payProject2(uint256 seed) external {
|
|
127
|
+
uint256 amount = bound(seed, 0.001 ether, 5 ether);
|
|
128
|
+
address actor = _getActor(seed);
|
|
129
|
+
vm.deal(actor, amount);
|
|
130
|
+
|
|
131
|
+
vm.prank(actor);
|
|
132
|
+
try terminal.pay{value: amount}(projectId2, JBConstants.NATIVE_TOKEN, amount, actor, 0, "", "") {
|
|
133
|
+
ghost_totalPaidIn[projectId2] += amount;
|
|
134
|
+
ghost_actorContributed[actor][projectId2] += amount;
|
|
135
|
+
ghost_globalInflows += amount;
|
|
136
|
+
callCount_pay2++;
|
|
137
|
+
} catch {}
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
/// @notice Pay into project 3.
|
|
141
|
+
function payProject3(uint256 seed) external {
|
|
142
|
+
uint256 amount = bound(seed, 0.001 ether, 5 ether);
|
|
143
|
+
address actor = _getActor(seed);
|
|
144
|
+
vm.deal(actor, amount);
|
|
145
|
+
|
|
146
|
+
vm.prank(actor);
|
|
147
|
+
try terminal.pay{value: amount}(projectId3, JBConstants.NATIVE_TOKEN, amount, actor, 0, "", "") {
|
|
148
|
+
ghost_totalPaidIn[projectId3] += amount;
|
|
149
|
+
ghost_actorContributed[actor][projectId3] += amount;
|
|
150
|
+
ghost_globalInflows += amount;
|
|
151
|
+
callCount_pay3++;
|
|
152
|
+
} catch {}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/// @notice Cash out tokens from project 2.
|
|
156
|
+
function cashOutProject2(uint256 seed) external {
|
|
157
|
+
address actor = _getActor(seed);
|
|
158
|
+
uint256 actorBalance = tokens.totalBalanceOf(actor, projectId2);
|
|
159
|
+
if (actorBalance == 0) return;
|
|
160
|
+
|
|
161
|
+
uint256 cashOutAmount = bound(seed, 1, actorBalance);
|
|
162
|
+
|
|
163
|
+
vm.prank(actor);
|
|
164
|
+
try terminal.cashOutTokensOf({
|
|
165
|
+
holder: actor,
|
|
166
|
+
projectId: projectId2,
|
|
167
|
+
cashOutCount: cashOutAmount,
|
|
168
|
+
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
169
|
+
minTokensReclaimed: 0,
|
|
170
|
+
beneficiary: payable(actor),
|
|
171
|
+
metadata: ""
|
|
172
|
+
}) returns (
|
|
173
|
+
uint256 reclaimAmount
|
|
174
|
+
) {
|
|
175
|
+
ghost_totalCashedOut[projectId2] += reclaimAmount;
|
|
176
|
+
ghost_actorExtracted[actor][projectId2] += reclaimAmount;
|
|
177
|
+
ghost_globalOutflows += reclaimAmount;
|
|
178
|
+
// Cash out fee is deducted from reclaim when cashOutTaxRate < MAX
|
|
179
|
+
uint256 feeAmount = _feeFrom(reclaimAmount);
|
|
180
|
+
ghost_totalFeesDeducted[projectId2] += feeAmount;
|
|
181
|
+
callCount_cashOut2++;
|
|
182
|
+
} catch {}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/// @notice Cash out tokens from project 3.
|
|
186
|
+
function cashOutProject3(uint256 seed) external {
|
|
187
|
+
address actor = _getActor(seed);
|
|
188
|
+
uint256 actorBalance = tokens.totalBalanceOf(actor, projectId3);
|
|
189
|
+
if (actorBalance == 0) return;
|
|
190
|
+
|
|
191
|
+
uint256 cashOutAmount = bound(seed, 1, actorBalance);
|
|
192
|
+
|
|
193
|
+
vm.prank(actor);
|
|
194
|
+
try terminal.cashOutTokensOf({
|
|
195
|
+
holder: actor,
|
|
196
|
+
projectId: projectId3,
|
|
197
|
+
cashOutCount: cashOutAmount,
|
|
198
|
+
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
199
|
+
minTokensReclaimed: 0,
|
|
200
|
+
beneficiary: payable(actor),
|
|
201
|
+
metadata: ""
|
|
202
|
+
}) returns (
|
|
203
|
+
uint256 reclaimAmount
|
|
204
|
+
) {
|
|
205
|
+
ghost_totalCashedOut[projectId3] += reclaimAmount;
|
|
206
|
+
ghost_actorExtracted[actor][projectId3] += reclaimAmount;
|
|
207
|
+
ghost_globalOutflows += reclaimAmount;
|
|
208
|
+
callCount_cashOut3++;
|
|
209
|
+
} catch {}
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
/// @notice Send payouts from project 2.
|
|
213
|
+
function sendPayoutsProject2(uint256 seed) external {
|
|
214
|
+
uint256 balance = store.balanceOf(address(terminal), projectId2, JBConstants.NATIVE_TOKEN);
|
|
215
|
+
if (balance == 0) return;
|
|
216
|
+
|
|
217
|
+
uint256 amount = bound(seed, 1, balance);
|
|
218
|
+
|
|
219
|
+
uint256 terminalBalBefore = address(terminal).balance;
|
|
220
|
+
vm.prank(projectOwner);
|
|
221
|
+
try terminal.sendPayoutsOf({
|
|
222
|
+
projectId: projectId2,
|
|
223
|
+
token: JBConstants.NATIVE_TOKEN,
|
|
224
|
+
amount: amount,
|
|
225
|
+
currency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
226
|
+
minTokensPaidOut: 0
|
|
227
|
+
}) returns (
|
|
228
|
+
uint256 amountPaidOut
|
|
229
|
+
) {
|
|
230
|
+
ghost_totalPaidOut[projectId2] += amountPaidOut;
|
|
231
|
+
// Track actual ETH that left the terminal, not the gross recorded amount.
|
|
232
|
+
// The fee stays in the terminal (paid to project #1 via internal _pay).
|
|
233
|
+
uint256 actualOutflow = terminalBalBefore - address(terminal).balance;
|
|
234
|
+
ghost_globalOutflows += actualOutflow;
|
|
235
|
+
callCount_sendPayouts2++;
|
|
236
|
+
} catch {}
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
/// @notice Use surplus allowance from project 2.
|
|
240
|
+
function useAllowanceProject2(uint256 seed) external {
|
|
241
|
+
uint256 balance = store.balanceOf(address(terminal), projectId2, JBConstants.NATIVE_TOKEN);
|
|
242
|
+
if (balance == 0) return;
|
|
243
|
+
|
|
244
|
+
uint256 amount = bound(seed, 1, balance > 3 ether ? 3 ether : balance);
|
|
245
|
+
|
|
246
|
+
vm.prank(projectOwner);
|
|
247
|
+
try terminal.useAllowanceOf({
|
|
248
|
+
projectId: projectId2,
|
|
249
|
+
token: JBConstants.NATIVE_TOKEN,
|
|
250
|
+
amount: amount,
|
|
251
|
+
currency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
252
|
+
minTokensPaidOut: 0,
|
|
253
|
+
beneficiary: payable(projectOwner),
|
|
254
|
+
feeBeneficiary: payable(projectOwner),
|
|
255
|
+
memo: ""
|
|
256
|
+
}) returns (
|
|
257
|
+
uint256 netAmountPaidOut
|
|
258
|
+
) {
|
|
259
|
+
ghost_totalAllowanceUsed[projectId2] += netAmountPaidOut;
|
|
260
|
+
ghost_globalOutflows += netAmountPaidOut;
|
|
261
|
+
callCount_useAllowance2++;
|
|
262
|
+
} catch {}
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
/// @notice Send reserved tokens for project 2.
|
|
266
|
+
function sendReservedTokens2(uint256 seed) external {
|
|
267
|
+
seed; // suppress unused warning
|
|
268
|
+
vm.prank(projectOwner);
|
|
269
|
+
try controller.sendReservedTokensToSplitsOf(projectId2) returns (uint256 tokenCount) {
|
|
270
|
+
ghost_totalReservesSent[projectId2] += tokenCount;
|
|
271
|
+
callCount_sendReserved2++;
|
|
272
|
+
} catch {}
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
/// @notice Process held fees for project 2.
|
|
276
|
+
function processHeldFees2(uint256 seed) external {
|
|
277
|
+
seed;
|
|
278
|
+
|
|
279
|
+
// Advance time past the 28-day holding period to ensure fees are unlocked
|
|
280
|
+
vm.warp(block.timestamp + 30 days);
|
|
281
|
+
|
|
282
|
+
// Count available held fees
|
|
283
|
+
JBFee[] memory fees = terminal.heldFeesOf(projectId2, JBConstants.NATIVE_TOKEN, 100);
|
|
284
|
+
if (fees.length == 0) return;
|
|
285
|
+
|
|
286
|
+
uint256 feeSum;
|
|
287
|
+
for (uint256 i = 0; i < fees.length; i++) {
|
|
288
|
+
feeSum += _feeFrom(fees[i].amount);
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
try terminal.processHeldFeesOf(projectId2, JBConstants.NATIVE_TOKEN, fees.length) {
|
|
292
|
+
ghost_totalProcessedFees[projectId2] += feeSum;
|
|
293
|
+
callCount_processHeldFees2++;
|
|
294
|
+
} catch {}
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
/// @notice Add to balance with fee return.
|
|
298
|
+
function addToBalanceReturnFees2(uint256 seed) external {
|
|
299
|
+
uint256 amount = bound(seed, 0.001 ether, 2 ether);
|
|
300
|
+
address actor = _getActor(seed);
|
|
301
|
+
vm.deal(actor, amount);
|
|
302
|
+
|
|
303
|
+
// Snapshot held fees before
|
|
304
|
+
JBFee[] memory feesBefore = terminal.heldFeesOf(projectId2, JBConstants.NATIVE_TOKEN, 100);
|
|
305
|
+
uint256 heldBefore;
|
|
306
|
+
for (uint256 i = 0; i < feesBefore.length; i++) {
|
|
307
|
+
heldBefore += feesBefore[i].amount;
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
vm.prank(actor);
|
|
311
|
+
try terminal.addToBalanceOf{value: amount}(
|
|
312
|
+
projectId2,
|
|
313
|
+
JBConstants.NATIVE_TOKEN,
|
|
314
|
+
amount,
|
|
315
|
+
true,
|
|
316
|
+
"",
|
|
317
|
+
"" // shouldReturnHeldFees = true
|
|
318
|
+
) {
|
|
319
|
+
ghost_totalAddedToBalance[projectId2] += amount;
|
|
320
|
+
ghost_globalInflows += amount;
|
|
321
|
+
|
|
322
|
+
// Track returned fees by measuring held fee change
|
|
323
|
+
JBFee[] memory feesAfter = terminal.heldFeesOf(projectId2, JBConstants.NATIVE_TOKEN, 100);
|
|
324
|
+
uint256 heldAfter;
|
|
325
|
+
for (uint256 i = 0; i < feesAfter.length; i++) {
|
|
326
|
+
heldAfter += feesAfter[i].amount;
|
|
327
|
+
}
|
|
328
|
+
if (heldBefore > heldAfter) {
|
|
329
|
+
ghost_totalReturnedFees[projectId2] += (heldBefore - heldAfter);
|
|
330
|
+
}
|
|
331
|
+
callCount_addToBalanceReturn2++;
|
|
332
|
+
} catch {}
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
/// @notice Add to balance without fee return.
|
|
336
|
+
function addToBalanceNoReturn2(uint256 seed) external {
|
|
337
|
+
uint256 amount = bound(seed, 0.001 ether, 2 ether);
|
|
338
|
+
address actor = _getActor(seed);
|
|
339
|
+
vm.deal(actor, amount);
|
|
340
|
+
|
|
341
|
+
vm.prank(actor);
|
|
342
|
+
try terminal.addToBalanceOf{value: amount}(
|
|
343
|
+
projectId2,
|
|
344
|
+
JBConstants.NATIVE_TOKEN,
|
|
345
|
+
amount,
|
|
346
|
+
false,
|
|
347
|
+
"",
|
|
348
|
+
"" // shouldReturnHeldFees = false
|
|
349
|
+
) {
|
|
350
|
+
ghost_totalAddedToBalance[projectId2] += amount;
|
|
351
|
+
ghost_globalInflows += amount;
|
|
352
|
+
callCount_addToBalanceNoReturn2++;
|
|
353
|
+
} catch {}
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
/// @notice Burn tokens for project 2.
|
|
357
|
+
function burnTokens2(uint256 seed) external {
|
|
358
|
+
address actor = _getActor(seed);
|
|
359
|
+
uint256 actorBalance = tokens.totalBalanceOf(actor, projectId2);
|
|
360
|
+
if (actorBalance == 0) return;
|
|
361
|
+
|
|
362
|
+
uint256 burnAmount = bound(seed, 1, actorBalance);
|
|
363
|
+
|
|
364
|
+
vm.prank(actor);
|
|
365
|
+
try controller.burnTokensOf({holder: actor, projectId: projectId2, tokenCount: burnAmount, memo: ""}) {
|
|
366
|
+
callCount_burnTokens2++;
|
|
367
|
+
} catch {}
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
/// @notice Burn tokens for project 3.
|
|
371
|
+
function burnTokens3(uint256 seed) external {
|
|
372
|
+
address actor = _getActor(seed);
|
|
373
|
+
uint256 actorBalance = tokens.totalBalanceOf(actor, projectId3);
|
|
374
|
+
if (actorBalance == 0) return;
|
|
375
|
+
|
|
376
|
+
uint256 burnAmount = bound(seed, 1, actorBalance);
|
|
377
|
+
|
|
378
|
+
vm.prank(actor);
|
|
379
|
+
try controller.burnTokensOf({holder: actor, projectId: projectId3, tokenCount: burnAmount, memo: ""}) {
|
|
380
|
+
callCount_burnTokens3++;
|
|
381
|
+
} catch {}
|
|
382
|
+
}
|
|
383
|
+
|
|
384
|
+
/// @notice Claim credits as ERC20 for project 2.
|
|
385
|
+
function claimCredits2(uint256 seed) external {
|
|
386
|
+
address actor = _getActor(seed);
|
|
387
|
+
|
|
388
|
+
// creditBalanceOf returns just the credit portion (not ERC20)
|
|
389
|
+
uint256 creditBalance = tokens.creditBalanceOf(actor, projectId2);
|
|
390
|
+
if (creditBalance == 0) return;
|
|
391
|
+
|
|
392
|
+
uint256 claimAmount = bound(seed, 1, creditBalance);
|
|
393
|
+
|
|
394
|
+
vm.prank(actor);
|
|
395
|
+
try tokens.claimTokensFor({holder: actor, projectId: projectId2, count: claimAmount, beneficiary: actor}) {
|
|
396
|
+
callCount_claimCredits2++;
|
|
397
|
+
} catch {}
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
/// @notice Advance block time.
|
|
401
|
+
function advanceTime(uint256 seed) external {
|
|
402
|
+
uint256 delta = bound(seed, 1 hours, 30 days);
|
|
403
|
+
vm.warp(block.timestamp + delta);
|
|
404
|
+
callCount_advanceTime++;
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
// =========================================================================
|
|
408
|
+
// View helpers for invariant checks
|
|
409
|
+
// =========================================================================
|
|
410
|
+
|
|
411
|
+
function getActor(uint256 index) external view returns (address) {
|
|
412
|
+
return actors[index];
|
|
413
|
+
}
|
|
414
|
+
}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
// SPDX-License-Identifier: MIT
|
|
2
|
+
pragma solidity ^0.8.6;
|
|
3
|
+
|
|
4
|
+
import "forge-std/Test.sol";
|
|
5
|
+
import {JBConstants} from "../../../src/libraries/JBConstants.sol";
|
|
6
|
+
import {IJBRulesets} from "../../../src/interfaces/IJBRulesets.sol";
|
|
7
|
+
import {IJBController} from "../../../src/interfaces/IJBController.sol";
|
|
8
|
+
import {IJBRulesetApprovalHook} from "../../../src/interfaces/IJBRulesetApprovalHook.sol";
|
|
9
|
+
import {JBRuleset} from "../../../src/structs/JBRuleset.sol";
|
|
10
|
+
import {JBRulesetMetadata} from "../../../src/structs/JBRulesetMetadata.sol";
|
|
11
|
+
import {JBRulesetConfig} from "../../../src/structs/JBRulesetConfig.sol";
|
|
12
|
+
import {JBSplitGroup} from "../../../src/structs/JBSplitGroup.sol";
|
|
13
|
+
import {JBFundAccessLimitGroup} from "../../../src/structs/JBFundAccessLimitGroup.sol";
|
|
14
|
+
import {JBRulesetMetadataResolver} from "../../../src/libraries/JBRulesetMetadataResolver.sol";
|
|
15
|
+
|
|
16
|
+
/// @notice Handler contract for JBRulesets invariant testing.
|
|
17
|
+
/// @dev Drives ruleset queueing, time advancement, and weight cache updates.
|
|
18
|
+
contract RulesetsHandler is Test {
|
|
19
|
+
using JBRulesetMetadataResolver for JBRulesetMetadata;
|
|
20
|
+
|
|
21
|
+
IJBRulesets public rulesets;
|
|
22
|
+
IJBController public controller;
|
|
23
|
+
|
|
24
|
+
uint256 public projectId;
|
|
25
|
+
address public projectOwner;
|
|
26
|
+
|
|
27
|
+
// Ghost: track the last observed cycle number and weight for monotonicity checks
|
|
28
|
+
uint48 public ghost_lastCycleNumber;
|
|
29
|
+
uint112 public ghost_lastWeight;
|
|
30
|
+
bool public ghost_hasLaunched;
|
|
31
|
+
uint256 public ghost_queueCount;
|
|
32
|
+
|
|
33
|
+
constructor(IJBRulesets _rulesets, IJBController _controller, uint256 _projectId, address _projectOwner) {
|
|
34
|
+
rulesets = _rulesets;
|
|
35
|
+
controller = _controller;
|
|
36
|
+
projectId = _projectId;
|
|
37
|
+
projectOwner = _projectOwner;
|
|
38
|
+
ghost_hasLaunched = true;
|
|
39
|
+
|
|
40
|
+
// Snapshot initial state
|
|
41
|
+
JBRuleset memory current = rulesets.currentOf(projectId);
|
|
42
|
+
ghost_lastCycleNumber = current.cycleNumber;
|
|
43
|
+
ghost_lastWeight = current.weight;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/// @notice Queue a new ruleset with random weight and duration.
|
|
47
|
+
function queueRuleset(uint256 weightSeed, uint256 durationSeed, uint256 weightCutSeed) public {
|
|
48
|
+
uint112 weight = uint112(bound(weightSeed, 1, 1e24));
|
|
49
|
+
uint32 duration = uint32(bound(durationSeed, 1 days, 365 days));
|
|
50
|
+
uint32 weightCutPercent = uint32(bound(weightCutSeed, 0, JBConstants.MAX_WEIGHT_CUT_PERCENT));
|
|
51
|
+
|
|
52
|
+
JBRulesetMetadata memory metadata = JBRulesetMetadata({
|
|
53
|
+
reservedPercent: 0,
|
|
54
|
+
cashOutTaxRate: 5000,
|
|
55
|
+
baseCurrency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
56
|
+
pausePay: false,
|
|
57
|
+
pauseCreditTransfers: false,
|
|
58
|
+
allowOwnerMinting: true,
|
|
59
|
+
allowSetCustomToken: true,
|
|
60
|
+
allowTerminalMigration: false,
|
|
61
|
+
allowSetTerminals: false,
|
|
62
|
+
ownerMustSendPayouts: false,
|
|
63
|
+
allowSetController: false,
|
|
64
|
+
allowAddAccountingContext: true,
|
|
65
|
+
allowAddPriceFeed: false,
|
|
66
|
+
holdFees: false,
|
|
67
|
+
useTotalSurplusForCashOuts: false,
|
|
68
|
+
useDataHookForPay: false,
|
|
69
|
+
useDataHookForCashOut: false,
|
|
70
|
+
dataHook: address(0),
|
|
71
|
+
metadata: 0
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
JBRulesetConfig[] memory configs = new JBRulesetConfig[](1);
|
|
75
|
+
configs[0] = JBRulesetConfig({
|
|
76
|
+
mustStartAtOrAfter: 0,
|
|
77
|
+
duration: duration,
|
|
78
|
+
weight: weight,
|
|
79
|
+
weightCutPercent: weightCutPercent,
|
|
80
|
+
approvalHook: IJBRulesetApprovalHook(address(0)),
|
|
81
|
+
metadata: metadata,
|
|
82
|
+
splitGroups: new JBSplitGroup[](0),
|
|
83
|
+
fundAccessLimitGroups: new JBFundAccessLimitGroup[](0)
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
vm.prank(projectOwner);
|
|
87
|
+
try controller.queueRulesetsOf({projectId: projectId, rulesetConfigurations: configs, memo: ""}) {
|
|
88
|
+
ghost_queueCount++;
|
|
89
|
+
} catch {
|
|
90
|
+
// May fail if not controller
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/// @notice Advance time to trigger ruleset cycling.
|
|
95
|
+
function advanceTime(uint256 timeSeed) public {
|
|
96
|
+
uint256 timeToAdvance = bound(timeSeed, 1 hours, 180 days);
|
|
97
|
+
vm.warp(block.timestamp + timeToAdvance);
|
|
98
|
+
|
|
99
|
+
// Update ghost state from current ruleset
|
|
100
|
+
JBRuleset memory current = rulesets.currentOf(projectId);
|
|
101
|
+
if (current.id != 0) {
|
|
102
|
+
ghost_lastCycleNumber = current.cycleNumber;
|
|
103
|
+
ghost_lastWeight = current.weight;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
/// @notice Update the weight cache for the project.
|
|
108
|
+
function updateWeightCache() public {
|
|
109
|
+
try rulesets.updateRulesetWeightCache(projectId) {} catch {}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
// SPDX-License-Identifier: MIT
|
|
2
|
+
pragma solidity ^0.8.6;
|
|
3
|
+
|
|
4
|
+
import "forge-std/Test.sol";
|
|
5
|
+
import {JBConstants} from "../../../src/libraries/JBConstants.sol";
|
|
6
|
+
import {IJBMultiTerminal} from "../../../src/interfaces/IJBMultiTerminal.sol";
|
|
7
|
+
import {IJBTerminalStore} from "../../../src/interfaces/IJBTerminalStore.sol";
|
|
8
|
+
import {IJBController} from "../../../src/interfaces/IJBController.sol";
|
|
9
|
+
import {IJBTokens} from "../../../src/interfaces/IJBTokens.sol";
|
|
10
|
+
|
|
11
|
+
/// @notice Handler contract for JBTerminalStore invariant testing.
|
|
12
|
+
/// @dev Wraps terminal operations and tracks ghost variables for conservation checks.
|
|
13
|
+
contract TerminalStoreHandler is Test {
|
|
14
|
+
IJBMultiTerminal public terminal;
|
|
15
|
+
IJBTerminalStore public store;
|
|
16
|
+
IJBController public controller;
|
|
17
|
+
IJBTokens public tokens;
|
|
18
|
+
|
|
19
|
+
uint256 public projectId;
|
|
20
|
+
address public projectOwner;
|
|
21
|
+
|
|
22
|
+
// Ghost variables for fund tracking
|
|
23
|
+
uint256 public ghost_totalPaidIn;
|
|
24
|
+
uint256 public ghost_totalCashedOut;
|
|
25
|
+
uint256 public ghost_totalPaidOut;
|
|
26
|
+
uint256 public ghost_totalAddedToBalance;
|
|
27
|
+
|
|
28
|
+
// Track actors
|
|
29
|
+
address[] public actors;
|
|
30
|
+
mapping(address => bool) public isActor;
|
|
31
|
+
uint256 public constant NUM_ACTORS = 5;
|
|
32
|
+
|
|
33
|
+
constructor(
|
|
34
|
+
IJBMultiTerminal _terminal,
|
|
35
|
+
IJBTerminalStore _store,
|
|
36
|
+
IJBController _controller,
|
|
37
|
+
IJBTokens _tokens,
|
|
38
|
+
uint256 _projectId,
|
|
39
|
+
address _projectOwner
|
|
40
|
+
) {
|
|
41
|
+
terminal = _terminal;
|
|
42
|
+
store = _store;
|
|
43
|
+
controller = _controller;
|
|
44
|
+
tokens = _tokens;
|
|
45
|
+
projectId = _projectId;
|
|
46
|
+
projectOwner = _projectOwner;
|
|
47
|
+
|
|
48
|
+
// Create actor addresses
|
|
49
|
+
for (uint256 i = 0; i < NUM_ACTORS; i++) {
|
|
50
|
+
address actor = address(uint160(0x1000 + i));
|
|
51
|
+
actors.push(actor);
|
|
52
|
+
isActor[actor] = true;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
/// @notice Selects an actor based on a seed.
|
|
57
|
+
function _getActor(uint256 seed) internal view returns (address) {
|
|
58
|
+
return actors[seed % actors.length];
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/// @notice Pay the project with native tokens.
|
|
62
|
+
function payProject(uint256 actorSeed, uint256 amount) public {
|
|
63
|
+
amount = bound(amount, 0.01 ether, 100 ether);
|
|
64
|
+
address actor = _getActor(actorSeed);
|
|
65
|
+
|
|
66
|
+
vm.deal(actor, amount);
|
|
67
|
+
vm.prank(actor);
|
|
68
|
+
terminal.pay{value: amount}({
|
|
69
|
+
projectId: projectId,
|
|
70
|
+
amount: amount,
|
|
71
|
+
token: JBConstants.NATIVE_TOKEN,
|
|
72
|
+
beneficiary: actor,
|
|
73
|
+
minReturnedTokens: 0,
|
|
74
|
+
memo: "",
|
|
75
|
+
metadata: new bytes(0)
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
ghost_totalPaidIn += amount;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/// @notice Cash out tokens for native tokens.
|
|
82
|
+
function cashOutTokens(uint256 actorSeed, uint256 cashOutPercent) public {
|
|
83
|
+
address actor = _getActor(actorSeed);
|
|
84
|
+
uint256 tokenBalance = tokens.totalBalanceOf(actor, projectId);
|
|
85
|
+
if (tokenBalance == 0) return;
|
|
86
|
+
|
|
87
|
+
cashOutPercent = bound(cashOutPercent, 1, 100);
|
|
88
|
+
uint256 cashOutCount = (tokenBalance * cashOutPercent) / 100;
|
|
89
|
+
if (cashOutCount == 0) return;
|
|
90
|
+
|
|
91
|
+
vm.prank(actor);
|
|
92
|
+
uint256 reclaimAmount = terminal.cashOutTokensOf({
|
|
93
|
+
holder: actor,
|
|
94
|
+
projectId: projectId,
|
|
95
|
+
cashOutCount: cashOutCount,
|
|
96
|
+
tokenToReclaim: JBConstants.NATIVE_TOKEN,
|
|
97
|
+
minTokensReclaimed: 0,
|
|
98
|
+
beneficiary: payable(actor),
|
|
99
|
+
metadata: new bytes(0)
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
ghost_totalCashedOut += reclaimAmount;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
/// @notice Send payouts from the project.
|
|
106
|
+
function sendPayouts(uint256 amount) public {
|
|
107
|
+
uint256 balance = store.balanceOf(address(terminal), projectId, JBConstants.NATIVE_TOKEN);
|
|
108
|
+
if (balance == 0) return;
|
|
109
|
+
|
|
110
|
+
amount = bound(amount, 1, balance);
|
|
111
|
+
|
|
112
|
+
vm.prank(projectOwner);
|
|
113
|
+
try terminal.sendPayoutsOf({
|
|
114
|
+
projectId: projectId,
|
|
115
|
+
token: JBConstants.NATIVE_TOKEN,
|
|
116
|
+
amount: amount,
|
|
117
|
+
currency: uint32(uint160(JBConstants.NATIVE_TOKEN)),
|
|
118
|
+
minTokensPaidOut: 0
|
|
119
|
+
}) returns (
|
|
120
|
+
uint256 amountPaidOut
|
|
121
|
+
) {
|
|
122
|
+
ghost_totalPaidOut += amountPaidOut;
|
|
123
|
+
} catch {
|
|
124
|
+
// Payout may fail if there's no payout limit configured
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/// @notice Add to project balance without minting tokens.
|
|
129
|
+
function addToBalance(uint256 amount) public {
|
|
130
|
+
amount = bound(amount, 0.01 ether, 50 ether);
|
|
131
|
+
|
|
132
|
+
vm.deal(address(this), amount);
|
|
133
|
+
terminal.addToBalanceOf{value: amount}({
|
|
134
|
+
projectId: projectId,
|
|
135
|
+
token: JBConstants.NATIVE_TOKEN,
|
|
136
|
+
amount: amount,
|
|
137
|
+
shouldReturnHeldFees: false,
|
|
138
|
+
memo: "",
|
|
139
|
+
metadata: new bytes(0)
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
ghost_totalAddedToBalance += amount;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
receive() external payable {}
|
|
146
|
+
}
|