From 2723b3d875d00ad377d74bfb3b64595dddda2ad5 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 26 Jun 2024 18:43:38 -0700 Subject: [PATCH 1/8] fix edit offer --- src/Strategy.sol | 68 +++++++++++++------ src/test/TestUSDCOffers.t.sol | 51 ++++++++------ src/test/mocks/MockTermAuctionOfferLocker.sol | 5 +- 3 files changed, 83 insertions(+), 41 deletions(-) diff --git a/src/Strategy.sol b/src/Strategy.sol index 4b1d225e..07c7b68e 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -1,7 +1,6 @@ // SPDX-License-Identifier: AGPL-3.0 pragma solidity 0.8.18; -import "forge-std/console.sol"; import {BaseStrategy, ERC20} from "@tokenized-strategy/BaseStrategy.sol"; import {SafeERC20, IERC20} from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import {IERC4626} from "@openzeppelin/contracts/interfaces/IERC4626.sol"; @@ -218,6 +217,16 @@ contract Strategy is BaseStrategy { _sweepAssetAndRedeemRepoTokens(0); } + function _generateOfferId( + bytes32 id, + address user, + address offerLocker + ) internal view returns (bytes32) { + return keccak256( + abi.encodePacked(id, user, offerLocker) + ); + } + function submitAuctionOffer( address termAuction, address repoToken, @@ -247,28 +256,38 @@ contract Strategy is BaseStrategy { _sweepAssetAndRedeemRepoTokens(0); //@dev sweep to ensure liquid balances up to date - uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); - uint256 offerAmount = RepoTokenUtils.purchaseToRepoPrecision( - repoTokenPrecision, PURCHASE_TOKEN_PRECISION, purchaseTokenAmount - ); uint256 liquidBalance = _totalLiquidBalance(address(this)); - - if (liquidBalance < purchaseTokenAmount) { - revert InsufficientLiquidBalance(liquidBalance, purchaseTokenAmount); + uint256 actualPurchaseTokenAmount = purchaseTokenAmount; + bytes32 offerId = _generateOfferId(idHash, address(this), address(offerLocker)); + uint256 currentOfferAmount = termAuctionListData.offers[offerId].offerAmount; + if (actualPurchaseTokenAmount > currentOfferAmount) { + unchecked { + actualPurchaseTokenAmount -= currentOfferAmount; + } } - uint256 resultingWeightedTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( - repoToken, offerAmount, liquidBalance - purchaseTokenAmount - ); - - if (resultingWeightedTimeToMaturity > timeToMaturityThreshold) { - revert TimeToMaturityAboveThreshold(); + if (liquidBalance < actualPurchaseTokenAmount) { + revert InsufficientLiquidBalance(liquidBalance, actualPurchaseTokenAmount); } - if ((liquidBalance - purchaseTokenAmount) < liquidityThreshold) { + if ((liquidBalance - actualPurchaseTokenAmount) < liquidityThreshold) { revert BalanceBelowLiquidityThreshold(); } + { + uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); + uint256 offerAmount = RepoTokenUtils.purchaseToRepoPrecision( + repoTokenPrecision, PURCHASE_TOKEN_PRECISION, purchaseTokenAmount + ); + uint256 resultingWeightedTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( + repoToken, offerAmount, liquidBalance - actualPurchaseTokenAmount + ); + + if (resultingWeightedTimeToMaturity > timeToMaturityThreshold) { + revert TimeToMaturityAboveThreshold(); + } + } + ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer; offer.id = idHash; @@ -277,14 +296,23 @@ contract Strategy is BaseStrategy { offer.amount = purchaseTokenAmount; offer.purchaseToken = address(asset); - offerIds = _submitOffer(auction, offerLocker, offer, repoToken); + offerIds = _submitOffer( + auction, + offerLocker, + offer, + repoToken, + actualPurchaseTokenAmount, + currentOfferAmount == 0 + ); } function _submitOffer( ITermAuction auction, ITermAuctionOfferLocker offerLocker, ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer, - address repoToken + address repoToken, + uint256 actualPurchaseTokenAmount, + bool newOffer ) private returns (bytes32[] memory offerIds) { ITermRepoServicer repoServicer = ITermRepoServicer(offerLocker.termRepoServicer()); @@ -292,15 +320,15 @@ contract Strategy is BaseStrategy { new ITermAuctionOfferLocker.TermAuctionOfferSubmission[](1); offerSubmissions[0] = offer; - _withdrawAsset(offer.amount); + _withdrawAsset(actualPurchaseTokenAmount); - ERC20(asset).approve(address(repoServicer.termRepoLocker()), offer.amount); + IERC20(asset).safeApprove(address(repoServicer.termRepoLocker()), actualPurchaseTokenAmount); offerIds = offerLocker.lockOffers(offerSubmissions); require(offerIds.length > 0); - if (termAuctionListData.offers[offerIds[0]].offerId == bytes32(0)) { + if (newOffer) { // new offer termAuctionListData.insertPending(PendingOffer({ offerId: offerIds[0], diff --git a/src/test/TestUSDCOffers.t.sol b/src/test/TestUSDCOffers.t.sol index c6800975..2e759967 100644 --- a/src/test/TestUSDCOffers.t.sol +++ b/src/test/TestUSDCOffers.t.sol @@ -1,6 +1,6 @@ pragma solidity ^0.8.18; -import "forge-std/console.sol"; +import "forge-std/console2.sol"; import {ERC20Mock} from "@openzeppelin/contracts/mocks/ERC20Mock.sol"; import {MockTermRepoToken} from "./mocks/MockTermRepoToken.sol"; import {MockTermAuction} from "./mocks/MockTermAuction.sol"; @@ -45,24 +45,25 @@ contract TestUSDCSubmitOffer is Setup { initialState.totalLiquidBalance = termStrategy.totalLiquidBalance(); } - function _submitOffer(bytes32 offerId, uint256 offerAmount) private { + function _submitOffer(bytes32 idHash, uint256 offerAmount) private returns (bytes32) { // test: only management can submit offers vm.expectRevert("!management"); bytes32[] memory offerIds = termStrategy.submitAuctionOffer( - address(repoToken1WeekAuction), address(repoToken1Week), offerId, bytes32("test price"), offerAmount + address(repoToken1WeekAuction), address(repoToken1Week), idHash, bytes32("test price"), offerAmount ); vm.prank(management); offerIds = termStrategy.submitAuctionOffer( - address(repoToken1WeekAuction), address(repoToken1Week), offerId, bytes32("test price"), offerAmount + address(repoToken1WeekAuction), address(repoToken1Week), idHash, bytes32("test price"), offerAmount ); assertEq(offerIds.length, 1); - assertEq(offerIds[0], offerId); + + return offerIds[0]; } function testSubmitOffer() public { - _submitOffer(bytes32("offer id 1"), 1e6); + _submitOffer(bytes32("offer id hash 1"), 1e6); assertEq(termStrategy.totalLiquidBalance(), initialState.totalLiquidBalance - 1e6); // test: totalAssetValue = total liquid balance + pending offer amount @@ -70,14 +71,15 @@ contract TestUSDCSubmitOffer is Setup { } function testEditOffer() public { - _submitOffer(bytes32("offer id 1"), 1e6); + bytes32 idHash1 = bytes32("offer id hash 1"); + bytes32 offerId1 = _submitOffer(idHash1, 1e6); // TODO: fuzz this uint256 offerAmount = 4e6; vm.prank(management); bytes32[] memory offerIds = termStrategy.submitAuctionOffer( - address(repoToken1WeekAuction), address(repoToken1Week), bytes32("offer id 1"), bytes32("test price"), offerAmount + address(repoToken1WeekAuction), address(repoToken1Week), idHash1, bytes32("test price"), offerAmount ); assertEq(termStrategy.totalLiquidBalance(), initialState.totalLiquidBalance - offerAmount); @@ -86,11 +88,11 @@ contract TestUSDCSubmitOffer is Setup { } function testDeleteOffers() public { - _submitOffer(bytes32("offer id 1"), 1e6); + bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), 1e6); bytes32[] memory offerIds = new bytes32[](1); - offerIds[0] = bytes32("offer id 1"); + offerIds[0] = offerId1; vm.expectRevert("!management"); termStrategy.deleteAuctionOffers(address(repoToken1WeekAuction), offerIds); @@ -126,10 +128,10 @@ contract TestUSDCSubmitOffer is Setup { } function testCompleteAuctionSuccessFull() public { - _submitOffer(bytes32("offer id 1"), 1e6); + bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), 1e6); bytes32[] memory offerIds = new bytes32[](1); - offerIds[0] = bytes32("offer id 1"); + offerIds[0] = offerId1; uint256[] memory fillAmounts = new uint256[](1); fillAmounts[0] = 1e6; uint256[] memory repoTokenAmounts = new uint256[](1); @@ -164,10 +166,10 @@ contract TestUSDCSubmitOffer is Setup { } function testCompleteAuctionSuccessPartial() public { - _submitOffer(bytes32("offer id 1"), 1e6); + bytes32 offerId1 = _submitOffer(bytes32("offer id 1"), 1e6); bytes32[] memory offerIds = new bytes32[](1); - offerIds[0] = bytes32("offer id 1"); + offerIds[0] = offerId1; uint256[] memory fillAmounts = new uint256[](1); // test: 50% filled @@ -204,7 +206,7 @@ contract TestUSDCSubmitOffer is Setup { } function testCompleteAuctionCanceled() public { - _submitOffer(bytes32("offer id 1"), 1e6); + bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), 1e6); repoToken1WeekAuction.auctionCanceled(); @@ -217,8 +219,8 @@ contract TestUSDCSubmitOffer is Setup { } function testMultipleOffers() public { - _submitOffer(bytes32("offer id 1"), 1e6); - _submitOffer(bytes32("offer id 2"), 5e6); + bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), 1e6); + bytes32 offerId2 = _submitOffer(bytes32("offer id hash 2"), 5e6); assertEq(termStrategy.totalLiquidBalance(), initialState.totalLiquidBalance - 6e6); // test: totalAssetValue = total liquid balance + pending offer amount @@ -227,7 +229,18 @@ contract TestUSDCSubmitOffer is Setup { bytes32[] memory offers = termStrategy.pendingOffers(); assertEq(offers.length, 2); - assertEq(offers[0], bytes32("offer id 2")); - assertEq(offers[1], bytes32("offer id 1")); + assertEq(offers[0], offerId2); + assertEq(offers[1], offerId1); + } + + function testEditOfferTotalGreaterThanCurrentLiquidity() public { + bytes32 idHash1 = bytes32("offer id hash 1"); + bytes32 offerId1 = _submitOffer(idHash1, 50e6); + + assertEq(termStrategy.totalLiquidBalance(), 50e6); + + _submitOffer(idHash1, 100e6); + + assertEq(termStrategy.totalLiquidBalance(), 0); } } diff --git a/src/test/mocks/MockTermAuctionOfferLocker.sol b/src/test/mocks/MockTermAuctionOfferLocker.sol index e749e05e..2c5213cc 100644 --- a/src/test/mocks/MockTermAuctionOfferLocker.sol +++ b/src/test/mocks/MockTermAuctionOfferLocker.sol @@ -55,7 +55,8 @@ contract MockTermAuctionOfferLocker is ITermAuctionOfferLocker { for (uint256 i; i < offerSubmissions.length; i++) { TermAuctionOfferSubmission memory submission = offerSubmissions[i]; - TermAuctionOffer memory offer = lockedOffers[submission.id]; + bytes32 offerId = keccak256(abi.encodePacked(submission.id,msg.sender,address(this))); + TermAuctionOffer memory offer = lockedOffers[offerId]; // existing offer if (offer.amount > 0) { @@ -68,7 +69,7 @@ contract MockTermAuctionOfferLocker is ITermAuctionOfferLocker { // update locked amount offer.amount = submission.amount; } else { - offer.id = submission.id; + offer.id = offerId; offer.offeror = submission.offeror; offer.offerPriceHash = submission.offerPriceHash; offer.amount = submission.amount; From 7168068aeb908acc1107164e4c1a47cca4dca807 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 26 Jun 2024 19:21:05 -0700 Subject: [PATCH 2/8] fixing edit offer --- src/Strategy.sol | 91 +++++++++++++++++++++-------------- src/test/TestUSDCOffers.t.sol | 11 +++++ 2 files changed, 67 insertions(+), 35 deletions(-) diff --git a/src/Strategy.sol b/src/Strategy.sol index 07c7b68e..4964b3c2 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -219,14 +219,31 @@ contract Strategy is BaseStrategy { function _generateOfferId( bytes32 id, - address user, address offerLocker ) internal view returns (bytes32) { return keccak256( - abi.encodePacked(id, user, offerLocker) + abi.encodePacked(id, address(this), offerLocker) ); } + function _validateWeightedMaturity( + address repoToken, + uint256 newOfferAmount, + uint256 newLiquidBalance + ) private { + uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); + uint256 offerAmountInRepoPrecision = RepoTokenUtils.purchaseToRepoPrecision( + repoTokenPrecision, PURCHASE_TOKEN_PRECISION, newOfferAmount + ); + uint256 resultingWeightedTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( + repoToken, offerAmountInRepoPrecision, newLiquidBalance + ); + + if (resultingWeightedTimeToMaturity > timeToMaturityThreshold) { + revert TimeToMaturityAboveThreshold(); + } + } + function submitAuctionOffer( address termAuction, address repoToken, @@ -257,35 +274,33 @@ contract Strategy is BaseStrategy { _sweepAssetAndRedeemRepoTokens(0); //@dev sweep to ensure liquid balances up to date uint256 liquidBalance = _totalLiquidBalance(address(this)); - uint256 actualPurchaseTokenAmount = purchaseTokenAmount; - bytes32 offerId = _generateOfferId(idHash, address(this), address(offerLocker)); + uint256 newOfferAmount = purchaseTokenAmount; + bytes32 offerId = _generateOfferId(idHash, address(offerLocker)); uint256 currentOfferAmount = termAuctionListData.offers[offerId].offerAmount; - if (actualPurchaseTokenAmount > currentOfferAmount) { + if (newOfferAmount > currentOfferAmount) { + uint256 offerDebit; unchecked { - actualPurchaseTokenAmount -= currentOfferAmount; + // checked above + offerDebit = newOfferAmount - currentOfferAmount; } - } - - if (liquidBalance < actualPurchaseTokenAmount) { - revert InsufficientLiquidBalance(liquidBalance, actualPurchaseTokenAmount); - } - - if ((liquidBalance - actualPurchaseTokenAmount) < liquidityThreshold) { - revert BalanceBelowLiquidityThreshold(); - } - - { - uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); - uint256 offerAmount = RepoTokenUtils.purchaseToRepoPrecision( - repoTokenPrecision, PURCHASE_TOKEN_PRECISION, purchaseTokenAmount - ); - uint256 resultingWeightedTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( - repoToken, offerAmount, liquidBalance - actualPurchaseTokenAmount - ); - - if (resultingWeightedTimeToMaturity > timeToMaturityThreshold) { - revert TimeToMaturityAboveThreshold(); + if (liquidBalance < offerDebit) { + revert InsufficientLiquidBalance(liquidBalance, offerDebit); + } + uint256 newLiquidBalance = liquidBalance - offerDebit; + if (newLiquidBalance < liquidityThreshold) { + revert BalanceBelowLiquidityThreshold(); + } + _validateWeightedMaturity(repoToken, newOfferAmount, newLiquidBalance); + } else { + uint256 offerCredit; + unchecked { + offerCredit = currentOfferAmount - newOfferAmount; + } + uint256 newLiquidBalance = liquidBalance + offerCredit; + if (newLiquidBalance < liquidityThreshold) { + revert BalanceBelowLiquidityThreshold(); } + _validateWeightedMaturity(repoToken, newOfferAmount, newLiquidBalance); } ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer; @@ -301,8 +316,8 @@ contract Strategy is BaseStrategy { offerLocker, offer, repoToken, - actualPurchaseTokenAmount, - currentOfferAmount == 0 + newOfferAmount, + currentOfferAmount ); } @@ -311,8 +326,8 @@ contract Strategy is BaseStrategy { ITermAuctionOfferLocker offerLocker, ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer, address repoToken, - uint256 actualPurchaseTokenAmount, - bool newOffer + uint256 newOfferAmount, + uint256 currentOfferAmount ) private returns (bytes32[] memory offerIds) { ITermRepoServicer repoServicer = ITermRepoServicer(offerLocker.termRepoServicer()); @@ -320,15 +335,21 @@ contract Strategy is BaseStrategy { new ITermAuctionOfferLocker.TermAuctionOfferSubmission[](1); offerSubmissions[0] = offer; - _withdrawAsset(actualPurchaseTokenAmount); - - IERC20(asset).safeApprove(address(repoServicer.termRepoLocker()), actualPurchaseTokenAmount); + if (newOfferAmount > currentOfferAmount) { + uint256 offerDebit; + unchecked { + // checked above + offerDebit = newOfferAmount - currentOfferAmount; + } + _withdrawAsset(offerDebit); + IERC20(asset).safeApprove(address(repoServicer.termRepoLocker()), offerDebit); + } offerIds = offerLocker.lockOffers(offerSubmissions); require(offerIds.length > 0); - if (newOffer) { + if (currentOfferAmount == 0) { // new offer termAuctionListData.insertPending(PendingOffer({ offerId: offerIds[0], diff --git a/src/test/TestUSDCOffers.t.sol b/src/test/TestUSDCOffers.t.sol index 2e759967..4374cb47 100644 --- a/src/test/TestUSDCOffers.t.sol +++ b/src/test/TestUSDCOffers.t.sol @@ -243,4 +243,15 @@ contract TestUSDCSubmitOffer is Setup { assertEq(termStrategy.totalLiquidBalance(), 0); } + + function testEditOfferTotalLessThanCurrentLiquidity() public { + bytes32 idHash1 = bytes32("offer id hash 1"); + bytes32 offerId1 = _submitOffer(idHash1, 100e6); + + assertEq(termStrategy.totalLiquidBalance(), 0); + + _submitOffer(idHash1, 50e6); + + assertEq(termStrategy.totalLiquidBalance(), 50e6); + } } From df9772658203b55f1f1437edd167f286b0cbfc41 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 26 Jun 2024 20:44:09 -0700 Subject: [PATCH 3/8] adding pending offers to weighted maturity math --- src/RepoTokenList.sol | 47 +++++++++++++------------- src/Strategy.sol | 50 +++++++++++++++++++++++----- src/TermAuctionList.sol | 47 +++++++++++++++++++++----- src/test/TestUSDCSellRepoToken.t.sol | 35 +++++++++++++++++++ 4 files changed, 138 insertions(+), 41 deletions(-) diff --git a/src/RepoTokenList.sol b/src/RepoTokenList.sol index 48735def..ffb60634 100644 --- a/src/RepoTokenList.sol +++ b/src/RepoTokenList.sol @@ -61,43 +61,50 @@ library RepoTokenList { } } - function simulateWeightedTimeToMaturity( - RepoTokenListData storage listData, + function getRepoTokenWeightedTimeToMaturity( + address repoToken, uint256 repoTokenBalanceInBaseAssetPrecision + ) internal view returns (uint256 weightedTimeToMaturity) { + uint256 currentMaturity = _getRepoTokenMaturity(repoToken); + + if (currentMaturity > block.timestamp) { + uint256 timeToMaturity = _getRepoTokenTimeToMaturity(currentMaturity, repoToken); + // Not matured yet + weightedTimeToMaturity = timeToMaturity * repoTokenBalanceInBaseAssetPrecision; + } + } + + function getCumulativeRepoTokenData( + RepoTokenListData storage listData, address repoToken, uint256 repoTokenAmount, uint256 purchaseTokenPrecision, uint256 liquidBalance - ) internal view returns (uint256) { - if (listData.head == NULL_NODE) return 0; + ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeRepoTokenAmount) { + if (listData.head == NULL_NODE) return (0, 0); - uint256 cumulativeWeightedTimeToMaturity; // in seconds - uint256 cumulativeRepoTokenAmount; // in purchase token precision address current = listData.head; bool found; while (current != NULL_NODE) { uint256 repoTokenBalance = ITermRepoToken(current).balanceOf(address(this)); if (repoTokenBalance > 0) { - uint256 redemptionValue = ITermRepoToken(current).redemptionValue(); - uint256 repoTokenPrecision = 10**ERC20(current).decimals(); - if (repoToken == current) { repoTokenBalance += repoTokenAmount; found = true; } + uint256 redemptionValue = ITermRepoToken(current).redemptionValue(); + uint256 repoTokenPrecision = 10**ERC20(current).decimals(); + uint256 repoTokenBalanceInBaseAssetPrecision = (redemptionValue * repoTokenBalance * purchaseTokenPrecision) / (repoTokenPrecision * RepoTokenUtils.RATE_PRECISION); - uint256 currentMaturity = _getRepoTokenMaturity(current); + uint256 weightedTimeToMaturity = getRepoTokenWeightedTimeToMaturity( + current, repoTokenBalanceInBaseAssetPrecision + ); - if (currentMaturity > block.timestamp) { - uint256 timeToMaturity = _getRepoTokenTimeToMaturity(currentMaturity, current); - // Not matured yet - cumulativeWeightedTimeToMaturity += - timeToMaturity * repoTokenBalanceInBaseAssetPrecision; - } + cumulativeWeightedTimeToMaturity += weightedTimeToMaturity; cumulativeRepoTokenAmount += repoTokenBalanceInBaseAssetPrecision; } @@ -120,14 +127,6 @@ library RepoTokenList { timeToMaturity * repoTokenAmountInBaseAssetPrecision; } } - - /// @dev avoid div by 0 - if (cumulativeRepoTokenAmount == 0 && liquidBalance == 0) { - return 0; - } - - // time * purchaseTokenPrecision / purchaseTokenPrecision - return cumulativeWeightedTimeToMaturity / (cumulativeRepoTokenAmount + liquidBalance); } function removeAndRedeemMaturedTokens(RepoTokenListData storage listData) internal { diff --git a/src/Strategy.sol b/src/Strategy.sol index 4964b3c2..c6e79707 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -84,14 +84,44 @@ contract Strategy is BaseStrategy { return termAuctionListData.pendingOffers(); } - function _removeRedeemAndCalculateWeightedMaturity( + function _calculateWeightedMaturity( address repoToken, uint256 amount, uint256 liquidBalance - ) private returns (uint256) { - return repoTokenListData.simulateWeightedTimeToMaturity( + ) private view returns (uint256) { + if ( + termAuctionListData.head == TermAuctionList.NULL_NODE && + repoTokenListData.head == RepoTokenList.NULL_NODE + ) return 0; + + uint256 cumulativeWeightedTimeToMaturity; // in seconds + uint256 cumulativeAmount; // in purchase token precision + + ( + uint256 cumulativeRepoTokenWeightedTimeToMaturity, + uint256 cumulativeRepoTokenAmount + ) = repoTokenListData.getCumulativeRepoTokenData( repoToken, amount, PURCHASE_TOKEN_PRECISION, liquidBalance ); + + cumulativeWeightedTimeToMaturity += cumulativeRepoTokenWeightedTimeToMaturity; + cumulativeAmount += cumulativeRepoTokenAmount; + + ( + uint256 cumulativeOfferWeightedTimeToMaturity, + uint256 cumulativeOfferAmount + ) = termAuctionListData.getCumulativeOfferData(repoTokenListData); + + cumulativeWeightedTimeToMaturity += cumulativeOfferWeightedTimeToMaturity; + cumulativeAmount += cumulativeOfferAmount; + + /// @dev avoid div by 0 + if (cumulativeAmount == 0 && liquidBalance == 0) { + return 0; + } + + // time * purchaseTokenPrecision / purchaseTokenPrecision + return cumulativeWeightedTimeToMaturity / (cumulativeAmount + liquidBalance); } function simulateWeightedTimeToMaturity(address repoToken, uint256 amount) external view returns (uint256) { @@ -99,9 +129,7 @@ contract Strategy is BaseStrategy { if (repoToken != address(0)) { repoTokenListData.validateRepoToken(ITermRepoToken(repoToken), termController, address(asset)); } - return repoTokenListData.simulateWeightedTimeToMaturity( - repoToken, amount, PURCHASE_TOKEN_PRECISION, _totalLiquidBalance(address(this)) - ); + return _calculateWeightedMaturity(repoToken, amount, _totalLiquidBalance(address(this))); } function calculateRepoTokenPresentValue( @@ -181,7 +209,7 @@ contract Strategy is BaseStrategy { revert InsufficientLiquidBalance(liquidBalance, proceeds); } - uint256 resultingTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( + uint256 resultingTimeToMaturity = _calculateWeightedMaturity( repoToken, repoTokenAmount, liquidBalance - proceeds ); @@ -235,7 +263,7 @@ contract Strategy is BaseStrategy { uint256 offerAmountInRepoPrecision = RepoTokenUtils.purchaseToRepoPrecision( repoTokenPrecision, PURCHASE_TOKEN_PRECISION, newOfferAmount ); - uint256 resultingWeightedTimeToMaturity = _removeRedeemAndCalculateWeightedMaturity( + uint256 resultingWeightedTimeToMaturity = _calculateWeightedMaturity( repoToken, offerAmountInRepoPrecision, newLiquidBalance ); @@ -278,6 +306,7 @@ contract Strategy is BaseStrategy { bytes32 offerId = _generateOfferId(idHash, address(offerLocker)); uint256 currentOfferAmount = termAuctionListData.offers[offerId].offerAmount; if (newOfferAmount > currentOfferAmount) { + // increasing offer amount uint256 offerDebit; unchecked { // checked above @@ -291,7 +320,8 @@ contract Strategy is BaseStrategy { revert BalanceBelowLiquidityThreshold(); } _validateWeightedMaturity(repoToken, newOfferAmount, newLiquidBalance); - } else { + } else if (currentOfferAmount > newOfferAmount) { + // decreasing offer amount uint256 offerCredit; unchecked { offerCredit = currentOfferAmount - newOfferAmount; @@ -301,6 +331,8 @@ contract Strategy is BaseStrategy { revert BalanceBelowLiquidityThreshold(); } _validateWeightedMaturity(repoToken, newOfferAmount, newLiquidBalance); + } else { + // no change in offer amount, do nothing } ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer; diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index 100d2f2a..b060da92 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -75,6 +75,13 @@ library TermAuctionList { ITermController termController, address asset ) internal { + /* + offer submitted; auction still open => include offerAmount in totalValue (otherwise locked purchaseToken will be missing from TV) + offer submitted; auction completed; !auctionClosed() => include offer.offerAmount in totalValue (because the offerLocker will have already deleted offer on completeAuction) + + even though repoToken has been transferred it hasn't been added to the repoTokenList + BUT only if it is new not a reopening + offer submitted; auction completed; auctionClosed() => repoToken has been added to the repoTokenList + */ if (listData.head == NULL_NODE) return; bytes32 current = listData.head; @@ -150,12 +157,36 @@ library TermAuctionList { current = _getNext(listData, current); } } -} -/* -offer submitted; auction still open => include offerAmount in totalValue (otherwise locked purchaseToken will be missing from TV) -offer submitted; auction completed; !auctionClosed() => include offer.offerAmount in totalValue (because the offerLocker will have already deleted offer on completeAuction) - + even though repoToken has been transferred it hasn't been added to the repoTokenList - BUT only if it is new not a reopening -offer submitted; auction completed; auctionClosed() => repoToken has been added to the repoTokenList -*/ \ No newline at end of file + function getCumulativeOfferData( + TermAuctionListData storage listData, + RepoTokenListData storage repoTokenListData + ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeOfferAmount) { + if (listData.head == NULL_NODE) return (0, 0); + + bytes32 current = listData.head; + while (current != NULL_NODE) { + PendingOffer memory offer = listData.offers[current]; + + uint256 offerAmount = offer.offerLocker.lockedOffer(offer.offerId).amount; + + /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up + /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings + if (offer.termAuction.auctionCompleted() && offerAmount == 0 && repoTokenListData.auctionRates[offer.repoToken] == 0) { + // set offerAmount to pending offer amount + offerAmount = offer.offerAmount; + } + + if (offerAmount > 0) { + uint256 weightedTimeToMaturity = RepoTokenList.getRepoTokenWeightedTimeToMaturity( + offer.repoToken, offerAmount + ); + + cumulativeWeightedTimeToMaturity += weightedTimeToMaturity; + cumulativeOfferAmount += offerAmount; + } + + current = _getNext(listData, current); + } + } +} diff --git a/src/test/TestUSDCSellRepoToken.t.sol b/src/test/TestUSDCSellRepoToken.t.sol index bbae1946..c18635f4 100644 --- a/src/test/TestUSDCSellRepoToken.t.sol +++ b/src/test/TestUSDCSellRepoToken.t.sol @@ -6,6 +6,7 @@ import {IERC4626} from "@openzeppelin/contracts/interfaces/IERC4626.sol"; import {ITokenizedStrategy} from "@tokenized-strategy/interfaces/ITokenizedStrategy.sol"; import {MockTermRepoToken} from "./mocks/MockTermRepoToken.sol"; import {MockTermController} from "./mocks/MockTermController.sol"; +import {MockTermAuction} from "./mocks/MockTermAuction.sol"; import {MockUSDC} from "./mocks/MockUSDC.sol"; import {Setup, ERC20, IStrategyInterface} from "./utils/Setup.sol"; import {ITermRepoToken} from "../interfaces/term/ITermRepoToken.sol"; @@ -180,6 +181,22 @@ contract TestUSDCSellRepoToken is Setup { _sellRepoTokens(tokens, amounts, false, true, ""); } + function _sell2RepoTokens( + MockTermRepoToken rt1, + uint256 amount1, + MockTermRepoToken rt2, + uint256 amount2 + ) private { + address[] memory tokens = new address[](2); + tokens[0] = address(rt1); + tokens[1] = address(rt2); + uint256[] memory amounts = new uint256[](2); + amounts[0] = amount1; + amounts[1] = amount2; + + _sellRepoTokens(tokens, amounts, false, true, ""); + } + function _sell3RepoTokensCheckHoldings() private { address[] memory holdings = termStrategy.repoTokenHoldings(); @@ -264,6 +281,24 @@ contract TestUSDCSellRepoToken is Setup { assertEq(termStrategy.simulateWeightedTimeToMaturity(address(0), 0), 1108800); } + // test: weighted maturity with both repo tokens and pending offers + function testSellMultipleRepoTokens_7_14_8_1_Offer_28_3() public { + _sell2RepoTokens(repoToken1Week, 8e18, repoToken2Week, 1e18); + + bytes32 idHash = bytes32("offer id hash 1"); + + MockTermAuction repoToken4WeekAuction = new MockTermAuction(repoToken4Week); + + mockUSDC.mint(address(termStrategy), 3e6); + + vm.prank(management); + termStrategy.submitAuctionOffer( + address(repoToken4WeekAuction), address(repoToken4Week), idHash, bytes32("test price"), 3e6 + ); + + assertEq(termStrategy.simulateWeightedTimeToMaturity(address(0), 0), 1108800); + } + function testSetGovernanceParameters() public { MockTermController newController = new MockTermController(); From 319460040abd9ee3de7387a0b341fbde1f4867fb Mon Sep 17 00:00:00 2001 From: 0xddong Date: Thu, 27 Jun 2024 08:24:34 -0700 Subject: [PATCH 4/8] fixing weighted time to maturity --- src/RepoTokenList.sol | 22 ++-------------------- src/Strategy.sol | 26 +++++++++++++++++++++----- src/TermAuctionList.sol | 14 ++++++++++---- 3 files changed, 33 insertions(+), 29 deletions(-) diff --git a/src/RepoTokenList.sol b/src/RepoTokenList.sol index ffb60634..5e7455d0 100644 --- a/src/RepoTokenList.sol +++ b/src/RepoTokenList.sol @@ -79,11 +79,10 @@ library RepoTokenList { uint256 repoTokenAmount, uint256 purchaseTokenPrecision, uint256 liquidBalance - ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeRepoTokenAmount) { - if (listData.head == NULL_NODE) return (0, 0); + ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeRepoTokenAmount, bool found) { + if (listData.head == NULL_NODE) return (0, 0, false); address current = listData.head; - bool found; while (current != NULL_NODE) { uint256 repoTokenBalance = ITermRepoToken(current).balanceOf(address(this)); @@ -110,23 +109,6 @@ library RepoTokenList { current = _getNext(listData, current); } - - /// @dev token is not found in the list (i.e. called from view function) - if (!found && repoToken != address(0)) { - uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); - uint256 redemptionValue = ITermRepoToken(repoToken).redemptionValue(); - uint256 repoTokenAmountInBaseAssetPrecision = - (redemptionValue * repoTokenAmount * purchaseTokenPrecision) / - (repoTokenPrecision * RepoTokenUtils.RATE_PRECISION); - - cumulativeRepoTokenAmount += repoTokenAmountInBaseAssetPrecision; - uint256 maturity = _getRepoTokenMaturity(repoToken); - if (maturity > block.timestamp) { - uint256 timeToMaturity = _getRepoTokenTimeToMaturity(maturity, repoToken); - cumulativeWeightedTimeToMaturity += - timeToMaturity * repoTokenAmountInBaseAssetPrecision; - } - } } function removeAndRedeemMaturedTokens(RepoTokenListData storage listData) internal { diff --git a/src/Strategy.sol b/src/Strategy.sol index c6e79707..121a85e3 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -86,7 +86,7 @@ contract Strategy is BaseStrategy { function _calculateWeightedMaturity( address repoToken, - uint256 amount, + uint256 repoTokenAmount, uint256 liquidBalance ) private view returns (uint256) { if ( @@ -99,9 +99,10 @@ contract Strategy is BaseStrategy { ( uint256 cumulativeRepoTokenWeightedTimeToMaturity, - uint256 cumulativeRepoTokenAmount + uint256 cumulativeRepoTokenAmount, + bool foundInRepoTokenList ) = repoTokenListData.getCumulativeRepoTokenData( - repoToken, amount, PURCHASE_TOKEN_PRECISION, liquidBalance + repoToken, repoTokenAmount, PURCHASE_TOKEN_PRECISION, liquidBalance ); cumulativeWeightedTimeToMaturity += cumulativeRepoTokenWeightedTimeToMaturity; @@ -109,12 +110,27 @@ contract Strategy is BaseStrategy { ( uint256 cumulativeOfferWeightedTimeToMaturity, - uint256 cumulativeOfferAmount - ) = termAuctionListData.getCumulativeOfferData(repoTokenListData); + uint256 cumulativeOfferAmount, + bool foundInOfferList + ) = termAuctionListData.getCumulativeOfferData(repoTokenListData, repoToken, repoTokenAmount); cumulativeWeightedTimeToMaturity += cumulativeOfferWeightedTimeToMaturity; cumulativeAmount += cumulativeOfferAmount; + if (!foundInRepoTokenList && !foundInOfferList && repoToken != address(0)) { + uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); + uint256 redemptionValue = ITermRepoToken(repoToken).redemptionValue(); + uint256 repoTokenAmountInBaseAssetPrecision = + (redemptionValue * repoTokenAmount * PURCHASE_TOKEN_PRECISION) / + (repoTokenPrecision * RepoTokenUtils.RATE_PRECISION); + + cumulativeAmount += repoTokenAmountInBaseAssetPrecision; + cumulativeWeightedTimeToMaturity += RepoTokenList.getRepoTokenWeightedTimeToMaturity( + repoToken, + repoTokenAmountInBaseAssetPrecision + ); + } + /// @dev avoid div by 0 if (cumulativeAmount == 0 && liquidBalance == 0) { return 0; diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index b060da92..aa642b11 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -160,15 +160,21 @@ library TermAuctionList { function getCumulativeOfferData( TermAuctionListData storage listData, - RepoTokenListData storage repoTokenListData - ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeOfferAmount) { - if (listData.head == NULL_NODE) return (0, 0); + RepoTokenListData storage repoTokenListData, + address repoToken, + uint256 newOfferAmount + ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeOfferAmount, bool found) { + if (listData.head == NULL_NODE) return (0, 0, false); bytes32 current = listData.head; while (current != NULL_NODE) { PendingOffer memory offer = listData.offers[current]; - uint256 offerAmount = offer.offerLocker.lockedOffer(offer.offerId).amount; + if (offer.repoToken == repoToken) { + found = true; + } + + uint256 offerAmount = offer.repoToken == repoToken ? newOfferAmount : offer.offerLocker.lockedOffer(offer.offerId).amount; /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings From 9011155ea2f63c166c061044bc3683b7a6f19902 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Thu, 27 Jun 2024 10:36:16 -0700 Subject: [PATCH 5/8] fix edit offer --- src/Strategy.sol | 7 +------ src/TermAuctionList.sol | 20 ++++++++++--------- src/test/mocks/MockTermAuctionOfferLocker.sol | 5 +++-- 3 files changed, 15 insertions(+), 17 deletions(-) diff --git a/src/Strategy.sol b/src/Strategy.sol index 121a85e3..e3446326 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -89,11 +89,6 @@ contract Strategy is BaseStrategy { uint256 repoTokenAmount, uint256 liquidBalance ) private view returns (uint256) { - if ( - termAuctionListData.head == TermAuctionList.NULL_NODE && - repoTokenListData.head == RepoTokenList.NULL_NODE - ) return 0; - uint256 cumulativeWeightedTimeToMaturity; // in seconds uint256 cumulativeAmount; // in purchase token precision @@ -353,7 +348,7 @@ contract Strategy is BaseStrategy { ITermAuctionOfferLocker.TermAuctionOfferSubmission memory offer; - offer.id = idHash; + offer.id = currentOfferAmount > 0 ? offerId : idHash; offer.offeror = address(this); offer.offerPriceHash = offerPriceHash; offer.amount = purchaseTokenAmount; diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index aa642b11..2f1bfe91 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -170,19 +170,21 @@ library TermAuctionList { while (current != NULL_NODE) { PendingOffer memory offer = listData.offers[current]; + uint256 offerAmount; if (offer.repoToken == repoToken) { + offerAmount = newOfferAmount; found = true; + } else { + offerAmount = offer.offerLocker.lockedOffer(offer.offerId).amount; + + /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up + /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings + if (offer.termAuction.auctionCompleted() && offerAmount == 0 && repoTokenListData.auctionRates[offer.repoToken] == 0) { + // set offerAmount to pending offer amount + offerAmount = offer.offerAmount; + } } - uint256 offerAmount = offer.repoToken == repoToken ? newOfferAmount : offer.offerLocker.lockedOffer(offer.offerId).amount; - - /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up - /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings - if (offer.termAuction.auctionCompleted() && offerAmount == 0 && repoTokenListData.auctionRates[offer.repoToken] == 0) { - // set offerAmount to pending offer amount - offerAmount = offer.offerAmount; - } - if (offerAmount > 0) { uint256 weightedTimeToMaturity = RepoTokenList.getRepoTokenWeightedTimeToMaturity( offer.repoToken, offerAmount diff --git a/src/test/mocks/MockTermAuctionOfferLocker.sol b/src/test/mocks/MockTermAuctionOfferLocker.sol index 2c5213cc..ff7e9f93 100644 --- a/src/test/mocks/MockTermAuctionOfferLocker.sol +++ b/src/test/mocks/MockTermAuctionOfferLocker.sol @@ -55,8 +55,7 @@ contract MockTermAuctionOfferLocker is ITermAuctionOfferLocker { for (uint256 i; i < offerSubmissions.length; i++) { TermAuctionOfferSubmission memory submission = offerSubmissions[i]; - bytes32 offerId = keccak256(abi.encodePacked(submission.id,msg.sender,address(this))); - TermAuctionOffer memory offer = lockedOffers[offerId]; + TermAuctionOffer memory offer = lockedOffers[submission.id]; // existing offer if (offer.amount > 0) { @@ -69,6 +68,8 @@ contract MockTermAuctionOfferLocker is ITermAuctionOfferLocker { // update locked amount offer.amount = submission.amount; } else { + bytes32 offerId = keccak256(abi.encodePacked(submission.id,msg.sender,address(this))); + offer.id = offerId; offer.offeror = submission.offeror; offer.offerPriceHash = submission.offerPriceHash; From 47d843d0d61d1d09a064972c313b5224b5c9fbf1 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 3 Jul 2024 17:59:11 -0700 Subject: [PATCH 6/8] fixing repo token PV math --- src/RepoTokenList.sol | 18 ++++++++--------- src/RepoTokenUtils.sol | 16 +++++++++++++++ src/Strategy.sol | 16 ++++++++------- src/TermAuctionList.sol | 37 +++++++++++++++++++++++++++++------ src/test/TestUSDCOffers.t.sol | 12 ++++++++---- 5 files changed, 73 insertions(+), 26 deletions(-) diff --git a/src/RepoTokenList.sol b/src/RepoTokenList.sol index 5e7455d0..41e2a7cd 100644 --- a/src/RepoTokenList.sol +++ b/src/RepoTokenList.sol @@ -27,7 +27,7 @@ library RepoTokenList { error InvalidRepoToken(address token); - function _getRepoTokenMaturity(address repoToken) private view returns (uint256 redemptionTimestamp) { + function getRepoTokenMaturity(address repoToken) internal view returns (uint256 redemptionTimestamp) { (redemptionTimestamp, , ,) = ITermRepoToken(repoToken).config(); } @@ -64,7 +64,7 @@ library RepoTokenList { function getRepoTokenWeightedTimeToMaturity( address repoToken, uint256 repoTokenBalanceInBaseAssetPrecision ) internal view returns (uint256 weightedTimeToMaturity) { - uint256 currentMaturity = _getRepoTokenMaturity(repoToken); + uint256 currentMaturity = getRepoTokenMaturity(repoToken); if (currentMaturity > block.timestamp) { uint256 timeToMaturity = _getRepoTokenTimeToMaturity(currentMaturity, repoToken); @@ -118,7 +118,7 @@ library RepoTokenList { address prev = current; while (current != NULL_NODE) { address next; - if (_getRepoTokenMaturity(current) < block.timestamp) { + if (getRepoTokenMaturity(current) < block.timestamp) { bool removeMaturedToken; uint256 repoTokenBalance = ITermRepoToken(current).balanceOf(address(this)); @@ -158,7 +158,7 @@ library RepoTokenList { } } - function _auctionRate(ITermController termController, ITermRepoToken repoToken) private view returns (uint256) { + function getAuctionRate(ITermController termController, ITermRepoToken repoToken) internal view returns (uint256) { (AuctionMetadata[] memory auctionMetadata, ) = termController.getTermAuctionResults(repoToken.termRepoId()); uint256 len = auctionMetadata.length; @@ -224,14 +224,14 @@ library RepoTokenList { revert InvalidRepoToken(address(repoToken)); } - uint256 oracleRate = _auctionRate(termController, repoToken); + uint256 oracleRate = getAuctionRate(termController, repoToken); if (oracleRate != INVALID_AUCTION_RATE) { if (auctionRate != oracleRate) { listData.auctionRates[address(repoToken)] = oracleRate; } } } else { - auctionRate = _auctionRate(termController, repoToken); + auctionRate = getAuctionRate(termController, repoToken); redemptionTimestamp = validateRepoToken(listData, repoToken, termController, asset); @@ -256,8 +256,8 @@ library RepoTokenList { break; } - uint256 currentMaturity = _getRepoTokenMaturity(current); - uint256 maturityToInsert = _getRepoTokenMaturity(repoToken); + uint256 currentMaturity = getRepoTokenMaturity(current); + uint256 maturityToInsert = getRepoTokenMaturity(repoToken); if (maturityToInsert <= currentMaturity) { if (prev == NULL_NODE) { @@ -289,7 +289,7 @@ library RepoTokenList { address current = listData.head; while (current != NULL_NODE) { - uint256 currentMaturity = _getRepoTokenMaturity(current); + uint256 currentMaturity = getRepoTokenMaturity(current); uint256 repoTokenBalance = ITermRepoToken(current).balanceOf(address(this)); uint256 repoTokenPrecision = 10**ERC20(current).decimals(); uint256 auctionRate = listData.auctionRates[current]; diff --git a/src/RepoTokenUtils.sol b/src/RepoTokenUtils.sol index 1e2fed9e..37653209 100644 --- a/src/RepoTokenUtils.sol +++ b/src/RepoTokenUtils.sol @@ -1,6 +1,9 @@ // SPDX-License-Identifier: AGPL-3.0 pragma solidity 0.8.18; +import {ERC20} from "@openzeppelin/contracts/token/ERC20/ERC20.sol"; +import {ITermRepoToken} from "./interfaces/term/ITermRepoToken.sol"; + library RepoTokenUtils { uint256 public constant THREESIXTY_DAYCOUNT_SECONDS = 360 days; uint256 public constant RATE_PRECISION = 1e18; @@ -35,4 +38,17 @@ library RepoTokenUtils { (repoTokenAmountInBaseAssetPrecision * purchaseTokenPrecision) / (purchaseTokenPrecision + (auctionRate * timeLeftToMaturityDayFraction / RATE_PRECISION)); } + + // returns repo token amount in base asset precision + function getNormalizedRepoTokenAmount( + address repoToken, + uint256 repoTokenAmount, + uint256 purchaseTokenPrecision + ) internal view returns (uint256 repoTokenAmountInBaseAssetPrecision) { + uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); + uint256 redemptionValue = ITermRepoToken(repoToken).redemptionValue(); + repoTokenAmountInBaseAssetPrecision = + (redemptionValue * repoTokenAmount * purchaseTokenPrecision) / + (repoTokenPrecision * RepoTokenUtils.RATE_PRECISION); + } } diff --git a/src/Strategy.sol b/src/Strategy.sol index e3446326..c81a0a57 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -107,17 +107,19 @@ contract Strategy is BaseStrategy { uint256 cumulativeOfferWeightedTimeToMaturity, uint256 cumulativeOfferAmount, bool foundInOfferList - ) = termAuctionListData.getCumulativeOfferData(repoTokenListData, repoToken, repoTokenAmount); + ) = termAuctionListData.getCumulativeOfferData( + repoTokenListData, termController, repoToken, repoTokenAmount, PURCHASE_TOKEN_PRECISION + ); cumulativeWeightedTimeToMaturity += cumulativeOfferWeightedTimeToMaturity; cumulativeAmount += cumulativeOfferAmount; if (!foundInRepoTokenList && !foundInOfferList && repoToken != address(0)) { - uint256 repoTokenPrecision = 10**ERC20(repoToken).decimals(); - uint256 redemptionValue = ITermRepoToken(repoToken).redemptionValue(); - uint256 repoTokenAmountInBaseAssetPrecision = - (redemptionValue * repoTokenAmount * PURCHASE_TOKEN_PRECISION) / - (repoTokenPrecision * RepoTokenUtils.RATE_PRECISION); + uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( + repoToken, + repoTokenAmount, + PURCHASE_TOKEN_PRECISION + ); cumulativeAmount += repoTokenAmountInBaseAssetPrecision; cumulativeWeightedTimeToMaturity += RepoTokenList.getRepoTokenWeightedTimeToMaturity( @@ -428,7 +430,7 @@ contract Strategy is BaseStrategy { function _totalAssetValue() internal view returns (uint256 totalValue) { return _totalLiquidBalance(address(this)) + repoTokenListData.getPresentValue(PURCHASE_TOKEN_PRECISION) + - termAuctionListData.getPresentValue(repoTokenListData); + termAuctionListData.getPresentValue(repoTokenListData, termController, PURCHASE_TOKEN_PRECISION); } constructor( diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index 2f1bfe91..3ac4a7bb 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -7,6 +7,7 @@ import {ITermController} from "./interfaces/term/ITermController.sol"; import {ITermRepoToken} from "./interfaces/term/ITermRepoToken.sol"; import {ERC20} from "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import {RepoTokenList, RepoTokenListData} from "./RepoTokenList.sol"; +import {RepoTokenUtils} from "./RepoTokenUtils.sol"; struct PendingOffer { bytes32 offerId; @@ -136,7 +137,9 @@ library TermAuctionList { function getPresentValue( TermAuctionListData storage listData, - RepoTokenListData storage repoTokenListData + RepoTokenListData storage repoTokenListData, + ITermController termController, + uint256 purchaseTokenPrecision ) internal view returns (uint256 totalValue) { if (listData.head == NULL_NODE) return 0; @@ -148,8 +151,18 @@ library TermAuctionList { /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings - if (offer.termAuction.auctionCompleted() && offerAmount == 0 && repoTokenListData.auctionRates[offer.repoToken] == 0) { - totalValue += offer.offerAmount; + if (offer.termAuction.auctionCompleted() && repoTokenListData.auctionRates[offer.repoToken] == 0) { + uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( + offer.repoToken, + ITermRepoToken(offer.repoToken).balanceOf(address(this)), + purchaseTokenPrecision + ); + totalValue += RepoTokenUtils.calculatePresentValue( + repoTokenAmountInBaseAssetPrecision, + purchaseTokenPrecision, + RepoTokenList.getRepoTokenMaturity(offer.repoToken), + RepoTokenList.getAuctionRate(termController, ITermRepoToken(offer.repoToken)) + ); } else { totalValue += offerAmount; } @@ -161,8 +174,10 @@ library TermAuctionList { function getCumulativeOfferData( TermAuctionListData storage listData, RepoTokenListData storage repoTokenListData, + ITermController termController, address repoToken, - uint256 newOfferAmount + uint256 newOfferAmount, + uint256 purchaseTokenPrecision ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeOfferAmount, bool found) { if (listData.head == NULL_NODE) return (0, 0, false); @@ -179,9 +194,19 @@ library TermAuctionList { /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings - if (offer.termAuction.auctionCompleted() && offerAmount == 0 && repoTokenListData.auctionRates[offer.repoToken] == 0) { + if (offer.termAuction.auctionCompleted() && repoTokenListData.auctionRates[offer.repoToken] == 0) { // set offerAmount to pending offer amount - offerAmount = offer.offerAmount; + uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( + offer.repoToken, + ITermRepoToken(offer.repoToken).balanceOf(address(this)), + purchaseTokenPrecision + ); + offerAmount = RepoTokenUtils.calculatePresentValue( + repoTokenAmountInBaseAssetPrecision, + purchaseTokenPrecision, + RepoTokenList.getRepoTokenMaturity(offer.repoToken), + RepoTokenList.getAuctionRate(termController, ITermRepoToken(offer.repoToken)) + ); } } diff --git a/src/test/TestUSDCOffers.t.sol b/src/test/TestUSDCOffers.t.sol index 4374cb47..eac062b0 100644 --- a/src/test/TestUSDCOffers.t.sol +++ b/src/test/TestUSDCOffers.t.sol @@ -129,18 +129,21 @@ contract TestUSDCSubmitOffer is Setup { function testCompleteAuctionSuccessFull() public { bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), 1e6); + uint256 fillAmount = 1e6; bytes32[] memory offerIds = new bytes32[](1); offerIds[0] = offerId1; uint256[] memory fillAmounts = new uint256[](1); - fillAmounts[0] = 1e6; + fillAmounts[0] = fillAmount; uint256[] memory repoTokenAmounts = new uint256[](1); repoTokenAmounts[0] = _getRepoTokenAmountGivenPurchaseTokenAmount( - 1e6, repoToken1Week, TEST_REPO_TOKEN_RATE + fillAmount, repoToken1Week, TEST_REPO_TOKEN_RATE ); repoToken1WeekAuction.auctionSuccess(offerIds, fillAmounts, repoTokenAmounts); + //console2.log("repoTokenAmounts[0]", repoTokenAmounts[0]); + // test: asset value should equal to initial asset value (liquid + pending offers) assertEq(termStrategy.totalAssetValue(), initialState.totalAssetValue); @@ -167,16 +170,17 @@ contract TestUSDCSubmitOffer is Setup { function testCompleteAuctionSuccessPartial() public { bytes32 offerId1 = _submitOffer(bytes32("offer id 1"), 1e6); + uint256 fillAmount = 0.5e6; bytes32[] memory offerIds = new bytes32[](1); offerIds[0] = offerId1; uint256[] memory fillAmounts = new uint256[](1); // test: 50% filled - fillAmounts[0] = 0.5e6; + fillAmounts[0] = fillAmount; uint256[] memory repoTokenAmounts = new uint256[](1); repoTokenAmounts[0] = _getRepoTokenAmountGivenPurchaseTokenAmount( - 0.5e6, repoToken1Week, TEST_REPO_TOKEN_RATE + fillAmount, repoToken1Week, TEST_REPO_TOKEN_RATE ); repoToken1WeekAuction.auctionSuccess(offerIds, fillAmounts, repoTokenAmounts); From 2e8dae0dd9a80e372ce23344f2c8d9d8535025b2 Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 3 Jul 2024 18:23:27 -0700 Subject: [PATCH 7/8] fix getCumulativeOfferData --- src/TermAuctionList.sol | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index 3ac4a7bb..4955ab91 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -195,18 +195,12 @@ library TermAuctionList { /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings if (offer.termAuction.auctionCompleted() && repoTokenListData.auctionRates[offer.repoToken] == 0) { - // set offerAmount to pending offer amount - uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( + // use normalized repo token amount if repo token is not in the list + offerAmount = RepoTokenUtils.getNormalizedRepoTokenAmount( offer.repoToken, ITermRepoToken(offer.repoToken).balanceOf(address(this)), purchaseTokenPrecision ); - offerAmount = RepoTokenUtils.calculatePresentValue( - repoTokenAmountInBaseAssetPrecision, - purchaseTokenPrecision, - RepoTokenList.getRepoTokenMaturity(offer.repoToken), - RepoTokenList.getAuctionRate(termController, ITermRepoToken(offer.repoToken)) - ); } } From 0070809000b9e21d56b5c979f36104c12ae1235d Mon Sep 17 00:00:00 2001 From: 0xddong Date: Wed, 3 Jul 2024 22:15:50 -0700 Subject: [PATCH 8/8] fixing repo token double counting issue --- src/Strategy.sol | 4 +- src/TermAuctionList.sol | 112 +++++++++++++++++++++++----------- src/test/TestUSDCOffers.t.sol | 26 ++++++++ 3 files changed, 105 insertions(+), 37 deletions(-) diff --git a/src/Strategy.sol b/src/Strategy.sol index c81a0a57..36f287d5 100644 --- a/src/Strategy.sol +++ b/src/Strategy.sol @@ -396,8 +396,7 @@ contract Strategy is BaseStrategy { if (currentOfferAmount == 0) { // new offer - termAuctionListData.insertPending(PendingOffer({ - offerId: offerIds[0], + termAuctionListData.insertPending(offerIds[0], PendingOffer({ repoToken: repoToken, offerAmount: offer.amount, termAuction: auction, @@ -406,7 +405,6 @@ contract Strategy is BaseStrategy { } else { // edit offer, overwrite existing termAuctionListData.offers[offerIds[0]] = PendingOffer({ - offerId: offerIds[0], repoToken: repoToken, offerAmount: offer.amount, termAuction: auction, diff --git a/src/TermAuctionList.sol b/src/TermAuctionList.sol index 4955ab91..0a87d899 100644 --- a/src/TermAuctionList.sol +++ b/src/TermAuctionList.sol @@ -10,11 +10,19 @@ import {RepoTokenList, RepoTokenListData} from "./RepoTokenList.sol"; import {RepoTokenUtils} from "./RepoTokenUtils.sol"; struct PendingOffer { + address repoToken; + uint256 offerAmount; + ITermAuction termAuction; + ITermAuctionOfferLocker offerLocker; +} + +struct PendingOfferMemory { bytes32 offerId; address repoToken; uint256 offerAmount; ITermAuction termAuction; ITermAuctionOfferLocker offerLocker; + bool isRepoTokenSeen; } struct TermAuctionListNode { @@ -58,16 +66,15 @@ library TermAuctionList { } } - function insertPending(TermAuctionListData storage listData, PendingOffer memory pendingOffer) internal { + function insertPending(TermAuctionListData storage listData, bytes32 offerId, PendingOffer memory pendingOffer) internal { bytes32 current = listData.head; - bytes32 id = pendingOffer.offerId; if (current != NULL_NODE) { - listData.nodes[id].next = current; + listData.nodes[offerId].next = current; } - listData.head = id; - listData.offers[id] = pendingOffer; + listData.head = offerId; + listData.offers[offerId] = pendingOffer; } function removeCompleted( @@ -91,7 +98,7 @@ library TermAuctionList { PendingOffer memory offer = listData.offers[current]; bytes32 next = _getNext(listData, current); - uint256 offerAmount = offer.offerLocker.lockedOffer(offer.offerId).amount; + uint256 offerAmount = offer.offerLocker.lockedOffer(current).amount; bool removeNode; bool insertRepoToken; @@ -111,7 +118,7 @@ library TermAuctionList { // withdraw manually bytes32[] memory offerIds = new bytes32[](1); - offerIds[0] = offer.offerId; + offerIds[0] = current; offer.offerLocker.unlockOffers(offerIds); } } @@ -135,6 +142,35 @@ library TermAuctionList { } } + function _loadOffers(TermAuctionListData storage listData) private view returns (PendingOfferMemory[] memory offers) { + uint256 len = _count(listData); + offers = new PendingOfferMemory[](len); + + uint256 i; + bytes32 current = listData.head; + while (current != NULL_NODE) { + PendingOffer memory currentOffer = listData.offers[current]; + PendingOfferMemory memory newOffer = offers[i]; + + newOffer.offerId = current; + newOffer.repoToken = currentOffer.repoToken; + newOffer.offerAmount = currentOffer.offerAmount; + newOffer.termAuction = currentOffer.termAuction; + newOffer.offerLocker = currentOffer.offerLocker; + + i++; + current = _getNext(listData, current); + } + } + + function _markRepoTokenAsSeen(PendingOfferMemory[] memory offers, address repoToken) private view { + for (uint256 i; i < offers.length; i++) { + if (repoToken == offers[i].repoToken) { + offers[i].isRepoTokenSeen = true; + } + } + } + function getPresentValue( TermAuctionListData storage listData, RepoTokenListData storage repoTokenListData, @@ -142,32 +178,37 @@ library TermAuctionList { uint256 purchaseTokenPrecision ) internal view returns (uint256 totalValue) { if (listData.head == NULL_NODE) return 0; + + PendingOfferMemory[] memory offers = _loadOffers(listData); - bytes32 current = listData.head; - while (current != NULL_NODE) { - PendingOffer memory offer = listData.offers[current]; + for (uint256 i; i < offers.length; i++) { + PendingOfferMemory memory offer = offers[i]; uint256 offerAmount = offer.offerLocker.lockedOffer(offer.offerId).amount; /// @dev offer processed, but auctionClosed not yet called and auction is new so repoToken not on List and wont be picked up /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings if (offer.termAuction.auctionCompleted() && repoTokenListData.auctionRates[offer.repoToken] == 0) { - uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( - offer.repoToken, - ITermRepoToken(offer.repoToken).balanceOf(address(this)), - purchaseTokenPrecision - ); - totalValue += RepoTokenUtils.calculatePresentValue( - repoTokenAmountInBaseAssetPrecision, - purchaseTokenPrecision, - RepoTokenList.getRepoTokenMaturity(offer.repoToken), - RepoTokenList.getAuctionRate(termController, ITermRepoToken(offer.repoToken)) - ); + if (!offer.isRepoTokenSeen) { + uint256 repoTokenAmountInBaseAssetPrecision = RepoTokenUtils.getNormalizedRepoTokenAmount( + offer.repoToken, + ITermRepoToken(offer.repoToken).balanceOf(address(this)), + purchaseTokenPrecision + ); + totalValue += RepoTokenUtils.calculatePresentValue( + repoTokenAmountInBaseAssetPrecision, + purchaseTokenPrecision, + RepoTokenList.getRepoTokenMaturity(offer.repoToken), + RepoTokenList.getAuctionRate(termController, ITermRepoToken(offer.repoToken)) + ); + + // since multiple offers can be tied to the same repo token, we need to mark + // the repo tokens we've seen to avoid double counting + _markRepoTokenAsSeen(offers, offer.repoToken); + } } else { totalValue += offerAmount; } - - current = _getNext(listData, current); } } @@ -181,9 +222,10 @@ library TermAuctionList { ) internal view returns (uint256 cumulativeWeightedTimeToMaturity, uint256 cumulativeOfferAmount, bool found) { if (listData.head == NULL_NODE) return (0, 0, false); - bytes32 current = listData.head; - while (current != NULL_NODE) { - PendingOffer memory offer = listData.offers[current]; + PendingOfferMemory[] memory offers = _loadOffers(listData); + + for (uint256 i; i < offers.length; i++) { + PendingOfferMemory memory offer = offers[i]; uint256 offerAmount; if (offer.repoToken == repoToken) { @@ -196,12 +238,16 @@ library TermAuctionList { /// checking repoTokenAuctionRates to make sure we are not double counting on re-openings if (offer.termAuction.auctionCompleted() && repoTokenListData.auctionRates[offer.repoToken] == 0) { // use normalized repo token amount if repo token is not in the list - offerAmount = RepoTokenUtils.getNormalizedRepoTokenAmount( - offer.repoToken, - ITermRepoToken(offer.repoToken).balanceOf(address(this)), - purchaseTokenPrecision - ); - } + if (!offer.isRepoTokenSeen) { + offerAmount = RepoTokenUtils.getNormalizedRepoTokenAmount( + offer.repoToken, + ITermRepoToken(offer.repoToken).balanceOf(address(this)), + purchaseTokenPrecision + ); + + _markRepoTokenAsSeen(offers, offer.repoToken); + } + } } if (offerAmount > 0) { @@ -212,8 +258,6 @@ library TermAuctionList { cumulativeWeightedTimeToMaturity += weightedTimeToMaturity; cumulativeOfferAmount += offerAmount; } - - current = _getNext(listData, current); } } } diff --git a/src/test/TestUSDCOffers.t.sol b/src/test/TestUSDCOffers.t.sol index eac062b0..1a11bfe7 100644 --- a/src/test/TestUSDCOffers.t.sol +++ b/src/test/TestUSDCOffers.t.sol @@ -237,6 +237,32 @@ contract TestUSDCSubmitOffer is Setup { assertEq(offers[1], offerId1); } + function testMultipleOffersFillAndNoFill() public { + uint256 offer1Amount = 1e6; + uint256 offer2Amount = 5e6; + bytes32 offerId1 = _submitOffer(bytes32("offer id hash 1"), offer1Amount); + bytes32 offerId2 = _submitOffer(bytes32("offer id hash 2"), offer2Amount); + + bytes32[] memory offerIds = new bytes32[](2); + offerIds[0] = offerId1; + offerIds[1] = offerId2; + uint256[] memory fillAmounts = new uint256[](2); + + // test: offer 1 filled, offer 2 not filled + fillAmounts[0] = offer1Amount; + fillAmounts[1] = 0; + uint256[] memory repoTokenAmounts = new uint256[](2); + repoTokenAmounts[0] = _getRepoTokenAmountGivenPurchaseTokenAmount( + offer1Amount, repoToken1Week, TEST_REPO_TOKEN_RATE + ); + repoTokenAmounts[1] = 0; + + repoToken1WeekAuction.auctionSuccess(offerIds, fillAmounts, repoTokenAmounts); + + // test: asset value should equal to initial asset value (liquid + pending offers) + assertEq(termStrategy.totalAssetValue(), initialState.totalAssetValue); + } + function testEditOfferTotalGreaterThanCurrentLiquidity() public { bytes32 idHash1 = bytes32("offer id hash 1"); bytes32 offerId1 = _submitOffer(idHash1, 50e6);