Conversation
… issue where redeem was not passing through the urchin.
Summary of Test Results if Merged To Main:
✅ All 115 tests passed! (0 skipped, Total: 115) Test Results for Merge
🔒 Security AnalysisHigh Severity Issuesarbitrary-send-ethImpact: AtomWallet._call(address,uint256,bytes) (src/AtomWallet.sol#214-221) sends eth to arbitrary user Dangerous calls: - (success,result) = target.call{value: value}(data) (src/AtomWallet.sol#215) Affected Files:
View Detailed Findings
Medium Severity IssuesView Medium Severity Issues##### incorrect-equality **Impact**: EthMultiVault._validateTimelock(bytes32) (src/EthMultiVault.sol#2214-2226) uses a dangerous strict equality: - timelock.readyTime == 0 (src/EthMultiVault.sol#2217)Affected Files:
reentrancy-no-ethImpact: Reentrancy in FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#219-241): External calls: - assets[i] = _redeem(shares[i],ids[i],curveIds[i]) (src/FeaUrchin.sol#232) - assets = ethMultiVault.redeemTriple(shares,address(this),id) (src/FeaUrchin.sol#145-147) - assets = ethMultiVault.redeemAtom(shares,address(this),id) (src/FeaUrchin.sol#145-147) - assets = ethMultiVault.redeemTripleCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#149-151) - assets = ethMultiVault.redeemAtomCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#149-151) State variables written after the call(s): - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalAssetsMoved += assets (src/FeaUrchin.sol#76) FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#24) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#24) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalAssetsStaked -= assets (src/FeaUrchin.sol#77) FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#25) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#25) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalFeesCollected += fee (src/FeaUrchin.sol#78) FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#26) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#26) Affected Files:
Recommended Actions
⛽ Gas Analysis📊 First gas snapshot created |
Summary of Test Results if Merged To Main:
✅ All 117 tests passed! (0 skipped, Total: 117) Test Results for Merge
🔒 Security AnalysisHigh Severity Issuesarbitrary-send-ethImpact: AtomWallet._call(address,uint256,bytes) (src/AtomWallet.sol#214-221) sends eth to arbitrary user Dangerous calls: - (success,result) = target.call{value: value}(data) (src/AtomWallet.sol#215) Affected Files:
View Detailed Findings
Medium Severity IssuesView Medium Severity Issues##### incorrect-equality **Impact**: EthMultiVault._validateTimelock(bytes32) (src/EthMultiVault.sol#2214-2226) uses a dangerous strict equality: - timelock.readyTime == 0 (src/EthMultiVault.sol#2217)Affected Files:
reentrancy-no-ethImpact: Reentrancy in FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#219-241): External calls: - assets[i] = _redeem(shares[i],ids[i],curveIds[i]) (src/FeaUrchin.sol#232) - assets = ethMultiVault.redeemTriple(shares,address(this),id) (src/FeaUrchin.sol#145-147) - assets = ethMultiVault.redeemAtom(shares,address(this),id) (src/FeaUrchin.sol#145-147) - assets = ethMultiVault.redeemTripleCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#149-151) - assets = ethMultiVault.redeemAtomCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#149-151) State variables written after the call(s): - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalAssetsMoved += assets (src/FeaUrchin.sol#76) FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#24) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#24) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalAssetsStaked -= assets (src/FeaUrchin.sol#77) FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#25) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#25) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#233) - totalFeesCollected += fee (src/FeaUrchin.sol#78) FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#26) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#66-72) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#74-80) - FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#26) Affected Files:
Recommended Actions
⛽ Gas Analysis |
Summary of Test Results if Merged To Main:
✅ All 117 tests passed! (0 skipped, Total: 117) Test Results for Merge
🔒 Security AnalysisHigh Severity Issuesarbitrary-send-ethImpact: AtomWallet._call(address,uint256,bytes) (src/AtomWallet.sol#214-221) sends eth to arbitrary user Dangerous calls: - (success,result) = target.call{value: value}(data) (src/AtomWallet.sol#215) Affected Files:
View Detailed Findings
Medium Severity IssuesView Medium Severity Issues##### incorrect-equality **Impact**: EthMultiVault._validateTimelock(bytes32) (src/EthMultiVault.sol#2214-2226) uses a dangerous strict equality: - timelock.readyTime == 0 (src/EthMultiVault.sol#2217)Affected Files:
reentrancy-no-ethImpact: Reentrancy in FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#367-389): External calls: - assets[i] = _redeem(shares[i],ids[i],curveIds[i]) (src/FeaUrchin.sol#380) - assets = ethMultiVault.redeemTriple(shares,address(this),id) (src/FeaUrchin.sol#267-269) - assets = ethMultiVault.redeemAtom(shares,address(this),id) (src/FeaUrchin.sol#267-269) - assets = ethMultiVault.redeemTripleCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#271-273) - assets = ethMultiVault.redeemAtomCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#271-273) State variables written after the call(s): - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#381) - totalAssetsMoved += assets (src/FeaUrchin.sol#165) FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#84) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#151-157) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#163-169) - FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#84) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#381) - totalAssetsStaked -= assets (src/FeaUrchin.sol#166) FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#86) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#151-157) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#163-169) - FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#86) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#381) - totalFeesCollected += fee (src/FeaUrchin.sol#167) FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#88) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#151-157) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#163-169) - FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#88) Affected Files:
Recommended Actions
⛽ Gas Analysis📊 First gas snapshot created |
Summary of Test Results if Merged To Main:
✅ All 120 tests passed! (0 skipped, Total: 120) Test Results for Merge
🔒 Security AnalysisHigh Severity Issuesarbitrary-send-ethImpact: AtomWallet._call(address,uint256,bytes) (src/AtomWallet.sol#214-221) sends eth to arbitrary user Dangerous calls: - (success,result) = target.call{value: value}(data) (src/AtomWallet.sol#215) Affected Files:
View Detailed Findings
reentrancy-ethImpact: Reentrancy in FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393): External calls: - shares[i] = _deposit(valuePerItem,receiver,ids[i],curveIds[i]) (src/FeaUrchin.sol#387) - shares = ethMultiVault.depositTriple{value: value}(receiver,id) (src/FeaUrchin.sol#245-247) - shares = ethMultiVault.depositAtom{value: value}(receiver,id) (src/FeaUrchin.sol#245-247) - shares = ethMultiVault.depositTripleCurve{value: value}(receiver,id,curveId) (src/FeaUrchin.sol#249-251) - shares = ethMultiVault.depositAtomCurve{value: value}(receiver,id,curveId) (src/FeaUrchin.sol#249-251) State variables written after the call(s): - userShares[receiver][ids[i]][curveIds[i]] += shares[i] (src/FeaUrchin.sol#388) FeaUrchin.userShares (src/FeaUrchin.sol#46) can be used in cross function reentrancies: - FeaUrchin.batchCreateAtom(bytes[]) (src/FeaUrchin.sol#340-348) - FeaUrchin.batchCreateTriple(uint256[],uint256[],uint256[]) (src/FeaUrchin.sol#355-367) - FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393) - FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424) - FeaUrchin.createAtom(bytes) (src/FeaUrchin.sol#193-199) - FeaUrchin.createTriple(uint256,uint256,uint256) (src/FeaUrchin.sol#206-217) - FeaUrchin.deposit(address,uint256,uint256) (src/FeaUrchin.sol#224-235) - FeaUrchin.getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#148-150) - FeaUrchin.redeem(uint256,address,uint256,uint256) (src/FeaUrchin.sol#270-285) Affected Files:
View Detailed Findings
Medium Severity IssuesView Medium Severity Issues##### incorrect-equality **Impact**: EthMultiVault._validateTimelock(bytes32) (src/EthMultiVault.sol#2219-2231) uses a dangerous strict equality: - timelock.readyTime == 0 (src/EthMultiVault.sol#2222)Affected Files:
reentrancy-no-ethImpact: Reentrancy in FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424): External calls: - assets[i] = _redeem(shares[i],ids[i],curveIds[i]) (src/FeaUrchin.sol#414) - assets = ethMultiVault.redeemTriple(shares,address(this),id) (src/FeaUrchin.sol#294-296) - assets = ethMultiVault.redeemAtom(shares,address(this),id) (src/FeaUrchin.sol#294-296) - assets = ethMultiVault.redeemTripleCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#298-300) - assets = ethMultiVault.redeemAtomCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#298-300) State variables written after the call(s): - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalAssetsMoved += assets (src/FeaUrchin.sol#184) FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#35) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#35) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalAssetsStaked -= assets (src/FeaUrchin.sol#185) FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#37) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#37) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalFeesCollected += fee (src/FeaUrchin.sol#186) FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#39) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#39) - userShares[receiver][ids[i]][curveIds[i]] -= shares[i] (src/FeaUrchin.sol#415) FeaUrchin.userShares (src/FeaUrchin.sol#46) can be used in cross function reentrancies: - FeaUrchin.batchCreateAtom(bytes[]) (src/FeaUrchin.sol#340-348) - FeaUrchin.batchCreateTriple(uint256[],uint256[],uint256[]) (src/FeaUrchin.sol#355-367) - FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393) - FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424) - FeaUrchin.createAtom(bytes) (src/FeaUrchin.sol#193-199) - FeaUrchin.createTriple(uint256,uint256,uint256) (src/FeaUrchin.sol#206-217) - FeaUrchin.deposit(address,uint256,uint256) (src/FeaUrchin.sol#224-235) - FeaUrchin.getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#148-150) - FeaUrchin.redeem(uint256,address,uint256,uint256) (src/FeaUrchin.sol#270-285) Affected Files:
unused-returnImpact: FeaUrchin._getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#140-146) ignores return value by (shares,None) = ethMultiVault.getVaultStateForUserCurve(termId,bondingCurveId,user) (src/FeaUrchin.sol#144) Affected Files:
Recommended Actions
⛽ Gas Analysis📊 First gas snapshot created |
Summary of Test Results if Merged To Main:
✅ All 124 tests passed! (0 skipped, Total: 124) Test Results for Merge
🔒 Security AnalysisHigh Severity Issuesarbitrary-send-ethImpact: AtomWallet._call(address,uint256,bytes) (src/AtomWallet.sol#214-221) sends eth to arbitrary user Dangerous calls: - (success,result) = target.call{value: value}(data) (src/AtomWallet.sol#215) Affected Files:
View Detailed Findings
reentrancy-ethImpact: Reentrancy in FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393): External calls: - shares[i] = _deposit(valuePerItem,receiver,ids[i],curveIds[i]) (src/FeaUrchin.sol#387) - shares = ethMultiVault.depositTriple{value: value}(receiver,id) (src/FeaUrchin.sol#245-247) - shares = ethMultiVault.depositAtom{value: value}(receiver,id) (src/FeaUrchin.sol#245-247) - shares = ethMultiVault.depositTripleCurve{value: value}(receiver,id,curveId) (src/FeaUrchin.sol#249-251) - shares = ethMultiVault.depositAtomCurve{value: value}(receiver,id,curveId) (src/FeaUrchin.sol#249-251) State variables written after the call(s): - userShares[receiver][ids[i]][curveIds[i]] += shares[i] (src/FeaUrchin.sol#388) FeaUrchin.userShares (src/FeaUrchin.sol#46) can be used in cross function reentrancies: - FeaUrchin.batchCreateAtom(bytes[]) (src/FeaUrchin.sol#340-348) - FeaUrchin.batchCreateTriple(uint256[],uint256[],uint256[]) (src/FeaUrchin.sol#355-367) - FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393) - FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424) - FeaUrchin.createAtom(bytes) (src/FeaUrchin.sol#193-199) - FeaUrchin.createTriple(uint256,uint256,uint256) (src/FeaUrchin.sol#206-217) - FeaUrchin.deposit(address,uint256,uint256) (src/FeaUrchin.sol#224-235) - FeaUrchin.getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#148-150) - FeaUrchin.redeem(uint256,address,uint256,uint256) (src/FeaUrchin.sol#270-285) Affected Files:
View Detailed Findings
Medium Severity IssuesView Medium Severity Issues##### incorrect-equality **Impact**: EthMultiVault._validateTimelock(bytes32) (src/EthMultiVault.sol#2187-2199) uses a dangerous strict equality: - timelock.readyTime == 0 (src/EthMultiVault.sol#2190)Affected Files:
reentrancy-no-ethImpact: Reentrancy in FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424): External calls: - assets[i] = _redeem(shares[i],ids[i],curveIds[i]) (src/FeaUrchin.sol#414) - assets = ethMultiVault.redeemTriple(shares,address(this),id) (src/FeaUrchin.sol#294-296) - assets = ethMultiVault.redeemAtom(shares,address(this),id) (src/FeaUrchin.sol#294-296) - assets = ethMultiVault.redeemTripleCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#298-300) - assets = ethMultiVault.redeemAtomCurve(shares,address(this),id,curveId) (src/FeaUrchin.sol#298-300) State variables written after the call(s): - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalAssetsMoved += assets (src/FeaUrchin.sol#184) FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#35) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalAssetsMoved (src/FeaUrchin.sol#35) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalAssetsStaked -= assets (src/FeaUrchin.sol#185) FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#37) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalAssetsStaked (src/FeaUrchin.sol#37) - (fee,netValue) = _processRedeem(assets[i]) (src/FeaUrchin.sol#416) - totalFeesCollected += fee (src/FeaUrchin.sol#186) FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#39) can be used in cross function reentrancies: - FeaUrchin._processDeposit(uint256) (src/FeaUrchin.sol#170-176) - FeaUrchin._processRedeem(uint256) (src/FeaUrchin.sol#182-188) - FeaUrchin.totalFeesCollected (src/FeaUrchin.sol#39) - userShares[receiver][ids[i]][curveIds[i]] -= shares[i] (src/FeaUrchin.sol#415) FeaUrchin.userShares (src/FeaUrchin.sol#46) can be used in cross function reentrancies: - FeaUrchin.batchCreateAtom(bytes[]) (src/FeaUrchin.sol#340-348) - FeaUrchin.batchCreateTriple(uint256[],uint256[],uint256[]) (src/FeaUrchin.sol#355-367) - FeaUrchin.batchDeposit(address,uint256[],uint256[]) (src/FeaUrchin.sol#374-393) - FeaUrchin.batchRedeem(uint256[],address,uint256[],uint256[]) (src/FeaUrchin.sol#401-424) - FeaUrchin.createAtom(bytes) (src/FeaUrchin.sol#193-199) - FeaUrchin.createTriple(uint256,uint256,uint256) (src/FeaUrchin.sol#206-217) - FeaUrchin.deposit(address,uint256,uint256) (src/FeaUrchin.sol#224-235) - FeaUrchin.getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#148-150) - FeaUrchin.redeem(uint256,address,uint256,uint256) (src/FeaUrchin.sol#270-285) Affected Files:
unused-returnImpact: FeaUrchin._getVaultShares(address,uint256,uint256) (src/FeaUrchin.sol#140-146) ignores return value by (shares,None) = ethMultiVault.getVaultStateForUserCurve(termId,bondingCurveId,user) (src/FeaUrchin.sol#144) Affected Files:
Recommended Actions
⛽ Gas Analysis📊 First gas snapshot created |
FeaUrchin Contract Implementation
Overview
This PR introduces the
FeaUrchincontract, a fee-taking wrapper around the EthMultiVault that adds a configurable fee structure for all vault operations. The implementation includes support for both standard and bonding curve operations, and provides an interface which is forward compatible by abstracting out "curve" functions, as well as term abstraction for atoms/triples. The purpose of this is for app developers to be able to charge fees, for coffee, or to fund their back-end wallets. There is a new template project in the monorepo which will certainly use this.Key Changes
New Contract: FeaUrchin
Interface Updates
Test Coverage
FeaUrchin.t.solImplementation Details
Fee Handling
fee = amount * feeNumerator / feeDenominatorAsset Tracking
totalAssetsMoved: Total value of assets that have moved through the contracttotalAssetsStaked: Current total value of assets stakedtotalFeesCollected: Total fees collected by the contractuniqueUsersCount: Number of unique users interacting with the contractBatch Operations
Testing
All core functionality has been tested, including:
Security Considerations
Commits
Note:
WE DO NOT WANT TO MERGE THIS YET BECAUSE THE MAIN BRANCH IS BEING AUDITED. This PR is for educational purposes, for now.