Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Preliminary Report fixes #158

Merged
merged 30 commits into from
Oct 29, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
1057387
fix: F-2024-6721 - added missing storage gap
Krishnakumarskr Oct 23, 2024
1ec48d6
fix: F-2024-6699 - changed the params order
Krishnakumarskr Oct 23, 2024
609d9b9
chore: Improve LiquidContinuousMultiTokenVault testing and branch cov…
lucasia Oct 23, 2024
2cbf196
chore: Improve LiquidContinuousMultiTokenVault testing and branch cov…
lucasia Oct 23, 2024
bcb1b71
chore: Improve LiquidContinuousMultiTokenVault testing
lucasia Oct 23, 2024
3c46905
chore: Improve LiquidContinuousMultiTokenVault testing - add convertT…
lucasia Oct 23, 2024
047ee0a
chore: Introduce support for different user accounts in MultiTokenVau…
lucasia Oct 24, 2024
e6d64fe
fix: Correct account verification (msgSender, owner, controller). Cl…
lucasia Oct 24, 2024
c3d58e0
fix: Correct account verification (msgSender, owner, controller). Cl…
lucasia Oct 24, 2024
6c28ce7
fix: F-2024-6693 revert if deposit is fractional
Krishnakumarskr Oct 25, 2024
8593697
:white_check_mark: Add Test Cases
nawar-hisso Oct 25, 2024
9117a32
:fire: Remove Comments
nawar-hisso Oct 25, 2024
227e4e3
chore: Utilize TestUsers in more tests. Remove duplicate loadTest code
lucasia Oct 25, 2024
769fee7
Tweaked to allow setting of a custom `maxDeposit` while also accessin…
jplodge-pro Oct 28, 2024
ef8d68b
Added the Branch Coverage option to the `genhtml` invocation.
jplodge-pro Oct 28, 2024
8288924
Added a test case for Max Deposit being exceeded on deposit.
jplodge-pro Oct 28, 2024
065ac0b
pulled latest changes
Krishnakumarskr Oct 28, 2024
48ec6c2
Merge branch 'preliminary-report-fixes' of https://github.com/credbul…
Krishnakumarskr Oct 28, 2024
9339f84
added test cases
Krishnakumarskr Oct 28, 2024
323511c
:twisted_rightwards_arrows: Merge
nawar-hisso Oct 28, 2024
0341366
:fire: Remove Unused Code
nawar-hisso Oct 28, 2024
aa503e5
Merge pull request #165 from credbull/optimize-multi-token-vault-test…
nawar-hisso Oct 28, 2024
4ad6226
test case for fractional redeem
Krishnakumarskr Oct 28, 2024
c616d7d
reverted fix for fractional shares and added tests
Krishnakumarskr Oct 28, 2024
a65e1e2
Merge branch 'preliminary-report-fixes' of https://github.com/credbul…
Krishnakumarskr Oct 28, 2024
0d24bb0
fix: Reduce LiquidStones minimum threshold for shares to assets conve…
lucasia Oct 28, 2024
ca9348c
chore: Fix minor linting issue
lucasia Oct 28, 2024
0258628
chore: Add test for Deposit and Redeem at cut-off times
lucasia Oct 28, 2024
c782b73
fix: Correct LiquidStone product to return 0 returns when depositing …
lucasia Oct 29, 2024
cce3896
F-2024-6700 - TimelockAsyncUnlock (#164)
ChaiSomsri96 Oct 29, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion packages/contracts/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
"dev": "yarn rm-dbdata && anvil --config-out localhost.json & make deploy-local",
"build": "forge build && yarn gen-types",
"test": "forge test",
"coverage": "forge coverage --report lcov && genhtml lcov.info -o out/test-reports/coverage --ignore-errors inconsistent",
"coverage": "forge coverage --report lcov && genhtml lcov.info --branch-coverage -o out/test-reports/coverage --ignore-errors inconsistent",
"format": "forge fmt && prettier './script/**/*.js' --write",
"lint": "forge fmt && eslint --fix --ignore-path .gitignore && yarn solhint './*(test|src)/**/*.sol'",
"db-check": "tsc && node ./script/utils/checkDb.js",
Expand Down
7 changes: 7 additions & 0 deletions packages/contracts/src/timelock/ITimelockAsyncUnlock.sol
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,13 @@ interface ITimelockAsyncUnlock {
external
returns (uint256[] memory depositPeriods, uint256[] memory amounts);

/**
* @notice Cancel a pending request to unlock
* @param owner Owner of the request
* @param requestId Discriminator between non-fungible requests
*/
function cancelRequestUnlock(address owner, uint256 requestId) external;

/**
* @dev Return notice period
*/
Expand Down
25 changes: 10 additions & 15 deletions packages/contracts/src/timelock/TimelockAsyncUnlock.sol
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,8 @@ abstract contract TimelockAsyncUnlock is Initializable, ITimelockAsyncUnlock, Co
// cache of user requested unlocks by depositPeriod across ALL requests. maps account => map(depositPeriod -> unlockAmount)
mapping(address account => EnumerableMap.UintToUintMap) private _depositPeriodAmountCache;

event CancelRedeemRequest(address indexed owner, uint256 indexed requestId, address indexed sender);

error TimelockAsyncUnlock__AuthorizeCallerFailed(address caller, address owner);
error TimelockAsyncUnlock__InvalidArrayLength(uint256 depositPeriodsLength, uint256 amountsLength);
error TimelockAsyncUnlock__ExceededMaxRequestUnlock(
Expand All @@ -33,7 +35,7 @@ abstract contract TimelockAsyncUnlock is Initializable, ITimelockAsyncUnlock, Co
error TimelockAsyncUnlock__UnlockBeforeDepositPeriod(
address caller, address owner, uint256 depositPeriod, uint256 unlockPeriod
);
error TimelockAsyncUnlock__UnlockBeforeUnlockPeriod(
error TimelockAsyncUnlock__UnlockBeforeCurrentPeriod(
address caller, address owner, uint256 currentPeriod, uint256 unlockPeriod
);

Expand Down Expand Up @@ -181,6 +183,10 @@ abstract contract TimelockAsyncUnlock is Initializable, ITimelockAsyncUnlock, Co
{
_authorizeCaller(_msgSender(), owner);

if (requestId > currentPeriod()) {
revert TimelockAsyncUnlock__UnlockBeforeCurrentPeriod(_msgSender(), owner, currentPeriod(), requestId);
}

// use copy of the depositPeriods and amounts. we will be altering the storage in _unlock()
(depositPeriods, amounts) = unlockRequests(owner, requestId);

Expand All @@ -200,7 +206,9 @@ abstract contract TimelockAsyncUnlock is Initializable, ITimelockAsyncUnlock, Co
internal
virtual
{
_handleUnlockValidation(owner, depositPeriod, requestId);
if (requestId < depositPeriod) {
revert TimelockAsyncUnlock__UnlockBeforeDepositPeriod(_msgSender(), owner, depositPeriod, requestId);
}

EnumerableMap.UintToUintMap storage unlockRequestsForRequestId = _unlockRequests[owner][requestId];

Expand Down Expand Up @@ -269,19 +277,6 @@ abstract contract TimelockAsyncUnlock is Initializable, ITimelockAsyncUnlock, Co
}
}

/**
* @dev An internal function to check if unlock can be performed
*/
function _handleUnlockValidation(address owner, uint256 depositPeriod, uint256 unlockPeriod) internal virtual {
if (unlockPeriod > currentPeriod()) {
revert TimelockAsyncUnlock__UnlockBeforeUnlockPeriod(_msgSender(), owner, currentPeriod(), unlockPeriod);
}

if (unlockPeriod < depositPeriod) {
revert TimelockAsyncUnlock__UnlockBeforeDepositPeriod(_msgSender(), owner, depositPeriod, unlockPeriod);
}
}

function supportsInterface(bytes4 interfaceId) public view virtual returns (bool) {
return interfaceId == type(ITimelockAsyncUnlock).interfaceId;
}
Expand Down
7 changes: 7 additions & 0 deletions packages/contracts/src/yield/CalcInterestMetadata.sol
Original file line number Diff line number Diff line change
Expand Up @@ -40,4 +40,11 @@ abstract contract CalcInterestMetadata is Initializable, ICalcInterestMetadata {
function scale() public view virtual returns (uint256 scale_) {
return SCALE;
}

/**
* @dev This empty reserved space is put in place to allow future versions to add new
* variables without shifting down storage in the inheritance chain.
* See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps
*/
uint256[49] private __gap;
}
96 changes: 75 additions & 21 deletions packages/contracts/src/yield/LiquidContinuousMultiTokenVault.sol
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,8 @@ contract LiquidContinuousMultiTokenVault is

error LiquidContinuousMultiTokenVault__InvalidFrequency(uint256 frequency);
error LiquidContinuousMultiTokenVault__InvalidAuthAddress(string authName, address authAddress);
error LiquidContinuousMultiTokenVault__ControllerMismatch(address sender, address controller);
error LiquidContinuousMultiTokenVault__ControllerNotSender(address sender, address controller);
error LiquidContinuousMultiTokenVault__UnAuthorized(address sender, address authorizedOwner);
error LiquidContinuousMultiTokenVault__AmountMismatch(uint256 amount1, uint256 amount2);
error LiquidContinuousMultiTokenVault__UnlockPeriodMismatch(uint256 unlockPeriod1, uint256 unlockPeriod2);
error LiquidContinuousMultiTokenVault__InvalidComponentTokenAmount(
Expand Down Expand Up @@ -125,7 +126,7 @@ contract LiquidContinuousMultiTokenVault is
override
returns (uint256 shares)
{
if (assets < SCALE) return 0; // no shares for fractional principal
if (assets < _minConversionThreshold()) return 0; // no shares for small fractional assets

return assets; // 1 asset = 1 share
}
Expand Down Expand Up @@ -161,7 +162,7 @@ contract LiquidContinuousMultiTokenVault is
override
returns (uint256 assets)
{
if (shares < SCALE) return 0; // no assets for fractional shares
if (shares < _minConversionThreshold()) return 0; // no assets for small fractional shares

if (redeemPeriod < depositPeriod) return 0; // trying to redeem before depositPeriod

Expand All @@ -179,11 +180,12 @@ contract LiquidContinuousMultiTokenVault is
* @param owner Source of the assets to deposit
* @return requestId_ Discriminator between non-fungible requests
*/
function requestDeposit(uint256 assets, address controller, address owner) public returns (uint256 requestId_) {
if (controller != _msgSender()) {
revert LiquidContinuousMultiTokenVault__ControllerMismatch(_msgSender(), controller);
}

function requestDeposit(uint256 assets, address controller, address owner)
public
onlyAuthorized(owner)
onlyController(controller)
returns (uint256 requestId_)
{
uint256 requestId = ZERO_REQUEST_ID; // requests and requestIds not used in buys.

deposit(assets, owner, controller);
Expand All @@ -196,11 +198,11 @@ contract LiquidContinuousMultiTokenVault is
* @param assets Amount of `asset` that was deposited by `requestDeposit`
* @param receiver Address to receive the shares
*/
function deposit(uint256 assets, address receiver, address controller) public returns (uint256 shares_) {
if (controller != _msgSender()) {
revert LiquidContinuousMultiTokenVault__ControllerMismatch(_msgSender(), controller);
}

function deposit(uint256 assets, address receiver, address controller)
public
onlyController(controller)
returns (uint256 shares_)
{
uint256 shares = deposit(assets, receiver);
emit Deposit(controller, receiver, assets, shares);
return shares;
Expand All @@ -212,42 +214,49 @@ contract LiquidContinuousMultiTokenVault is
* @param owner Source of the shares to redeem
* @return requestId_ Discriminator between non-fungible requests
*/
function requestRedeem(uint256 shares, address, /* controller */ address owner)
function requestRedeem(uint256 shares, address controller, address owner)
public
onlyAuthorized(owner)
onlyController(controller)
returns (uint256 requestId_)
{
// using optimize() variant in case "shares" represents the IComponent "principal + yield" which is our "assets".
(uint256[] memory depositPeriods, uint256[] memory sharesAtPeriods) =
_redeemOptimizer.optimize(this, owner, shares, shares, minUnlockPeriod());

uint256 requestId = requestUnlock(_msgSender(), depositPeriods, sharesAtPeriods);
emit RedeemRequest(_msgSender(), owner, requestId, _msgSender(), shares);
uint256 requestId = requestUnlock(owner, depositPeriods, sharesAtPeriods);
emit RedeemRequest(controller, owner, requestId, _msgSender(), shares);
return requestId;
}

/**
* @notice Fulfill a request to redeem assets by transferring assets to the receiver
* @param shares Amount of shares that was redeemed by `requestRedeem`
* @param receiver Address to receive the assets
* @dev controller will only have tokens to redeem if they are also the owner
*/
function redeem(uint256 shares, address receiver, address /* controller */ ) public returns (uint256 assets) {
function redeem(uint256 shares, address receiver, address controller)
public
onlyController(controller)
returns (uint256 assets)
{
uint256 requestId = currentPeriod(); // requestId = redeemPeriod, and redeem can only be called where redeemPeriod = currentPeriod()

uint256 unlockRequestedAmount = unlockRequestAmount(receiver, requestId);
uint256 unlockRequestedAmount = unlockRequestAmount(controller, requestId);
if (shares != unlockRequestedAmount) {
revert LiquidContinuousMultiTokenVault__InvalidComponentTokenAmount(shares, unlockRequestedAmount);
}

(uint256[] memory depositPeriods, uint256[] memory sharesAtPeriods) = unlock(receiver, requestId); // unlockPeriod = redeemPeriod
(uint256[] memory depositPeriods, uint256[] memory sharesAtPeriods) = unlock(controller, requestId); // unlockPeriod = redeemPeriod

uint256 totalAssetsRedeemed = 0;

for (uint256 i = 0; i < depositPeriods.length; ++i) {
totalAssetsRedeemed += _redeemForDepositPeriodAfterUnlock(
sharesAtPeriods[i], receiver, _msgSender(), depositPeriods[i], requestId
sharesAtPeriods[i], receiver, controller, depositPeriods[i], requestId
);
}
emit Withdraw(_msgSender(), receiver, _msgSender(), totalAssetsRedeemed, shares);
emit Withdraw(_msgSender(), receiver, controller, totalAssetsRedeemed, shares);
return totalAssetsRedeemed;
}

Expand Down Expand Up @@ -367,6 +376,12 @@ contract LiquidContinuousMultiTokenVault is

/// @dev yield based on the associated yieldStrategy
function calcYield(uint256 principal, uint256 fromPeriod, uint256 toPeriod) public view returns (uint256 yield) {
// no yield earned when depositing and requesting redeem within the notice period.
// e.g. deposit day 1, immediately request redeem on day 1. should give 0 returns.
if (toPeriod <= fromPeriod + noticePeriod()) {
return 0;
}

return _yieldStrategy.calcYield(address(this), principal, fromPeriod, toPeriod);
}

Expand All @@ -383,6 +398,17 @@ contract LiquidContinuousMultiTokenVault is
_depositForDepositPeriod(amount, account, depositPeriod);
}

/// @dev Cancel a pending request to unlock
function cancelRequestUnlock(address owner, uint256 requestId) public onlyAuthorized(owner) {
(uint256[] memory depositPeriods, uint256[] memory amounts) = unlockRequests(owner, requestId);

for (uint256 i = 0; i < depositPeriods.length; ++i) {
_unlock(owner, depositPeriods[i], requestId, amounts[i]);
}

emit CancelRedeemRequest(owner, requestId, _msgSender());
}

/// @inheritdoc TimelockAsyncUnlock
function lockedAmount(address account, uint256 depositPeriod)
public
Expand All @@ -393,6 +419,13 @@ contract LiquidContinuousMultiTokenVault is
return balanceOf(account, depositPeriod);
}

/// @inheritdoc TimelockAsyncUnlock
function _authorizeCaller(address caller, address owner) internal virtual override {
if (caller != owner && !isApprovedForAll(owner, caller)) {
revert LiquidContinuousMultiTokenVault__UnAuthorized(caller, owner);
}
}

// ===================== TripleRateContext =====================

/// @inheritdoc TripleRateContext
Expand Down Expand Up @@ -458,6 +491,27 @@ contract LiquidContinuousMultiTokenVault is

// ===================== Utility =====================

/// minimum shares required to convert to assets and vice-versa.
function _minConversionThreshold() internal view returns (uint256 minConversionThreshold) {
return SCALE < 10 ? SCALE : 10;
}

// @dev ensure caller is permitted to act on the owner's tokens
modifier onlyAuthorized(address owner) {
_authorizeCaller(_msgSender(), owner);
_;
}

// @dev ensure the controller is the caller
modifier onlyController(address controller) {
address caller = _msgSender();

if (caller != controller) {
revert LiquidContinuousMultiTokenVault__ControllerNotSender(caller, controller);
}
_;
}

function pause() public onlyRole(OPERATOR_ROLE) {
_pause();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ contract TripleRateYieldStrategy is AbstractYieldStrategy {
ITripleRateContext context = ITripleRateContext(contextContract);

return CalcSimpleInterest.calcPriceFromInterest(
numPeriodsElapsed, context.rateScaled(), context.frequency(), context.scale()
context.rateScaled(), numPeriodsElapsed, context.frequency(), context.scale()
);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -185,7 +185,7 @@ contract TimelockAsyncUnlockTest is Test {

vm.expectRevert(
abi.encodeWithSelector(
TimelockAsyncUnlock.TimelockAsyncUnlock__UnlockBeforeUnlockPeriod.selector,
TimelockAsyncUnlock.TimelockAsyncUnlock__UnlockBeforeCurrentPeriod.selector,
alice,
alice,
asyncUnlock.currentPeriod(),
Expand Down
Loading
Loading