From 34704753a783fe54874cc1c073928447d10faa3f Mon Sep 17 00:00:00 2001 From: Chris Hibbert Date: Mon, 15 May 2023 21:03:50 -0700 Subject: [PATCH 1/2] fix(vaults): when reconstituting vaults, report correct shortfall fixes: #7474 We weren't reducing the reported shortfall by the amount of debt being restored to some vaults. --- .../src/vaultFactory/vaultManager.js | 19 ++++++++++++------- .../vaultFactory/test-vaultLiquidation.js | 2 +- 2 files changed, 13 insertions(+), 8 deletions(-) diff --git a/packages/inter-protocol/src/vaultFactory/vaultManager.js b/packages/inter-protocol/src/vaultFactory/vaultManager.js index dd3274a9aab..7d2a56261a6 100644 --- a/packages/inter-protocol/src/vaultFactory/vaultManager.js +++ b/packages/inter-protocol/src/vaultFactory/vaultManager.js @@ -715,6 +715,7 @@ export const prepareVaultManagerKit = ( vaultData.getSize(), ); + facets.helper.recordShortfallAndProceeds(accounting); facets.helper.markDoneLiquidating(totalDebt, totalCollateral); } else if (AmountMath.isEmpty(collateralProceeds)) { // Flow #2a @@ -782,6 +783,7 @@ export const prepareVaultManagerKit = ( vaultData.getSize(), ); + facets.helper.recordShortfallAndProceeds(accounting); facets.helper.markDoneLiquidating(totalDebt, totalCollateral); } else { // Flow #2b: There's unsold collateral; some vaults may be revived. @@ -801,13 +803,13 @@ export const prepareVaultManagerKit = ( : AmountMath.makeEmptyFromAmount(collateralProceeds); let collatRemaining = distributableCollateral; - let debtRemaining = totalDebt; /** @type {import('@agoric/zoe/src/contractSupport/atomicTransfer.js').TransferPart[]} */ const transfers = []; let liquidated = 0; /** @type {MapStore} */ const vaultsToReinstate = makeScalarMapStore(); let collateralReduction = AmountMath.makeEmpty(collateralBrand); + let shortfallToReserve = accounting.shortfall; const reduceCollateral = amount => (collateralReduction = AmountMath.add( @@ -825,13 +827,16 @@ export const prepareVaultManagerKit = ( if ( reconstituteVaults && AmountMath.isGTE(collatRemaining, collatPostDebt) && - AmountMath.isGTE(debtRemaining, debtAmount) + AmountMath.isGTE(totalDebt, debtAmount) ) { collatRemaining = AmountMath.subtract( collatRemaining, collatPostDebt, ); - debtRemaining = AmountMath.subtract(debtRemaining, debtAmount); + shortfallToReserve = AmountMath.subtract( + shortfallToReserve, + debtAmount, + ); const seat = vault.getVaultSeat(); const vaultId = vault.abortLiquidation(); liquidatingVaults.delete(vault); @@ -869,13 +874,13 @@ export const prepareVaultManagerKit = ( transfers.length, ); - facets.helper.markRestoreDebt( - AmountMath.subtract(totalDebt, debtRemaining), - ); facets.helper.sendToReserve(collatRemaining, liqSeat); facets.helper.markDoneLiquidating(totalDebt, totalCollateral); + facets.helper.recordShortfallAndProceeds({ + overage: accounting.overage, + shortfall: shortfallToReserve, + }); } - facets.helper.recordShortfallAndProceeds(accounting); return facets.helper.updateMetrics(); }, }, diff --git a/packages/inter-protocol/test/vaultFactory/test-vaultLiquidation.js b/packages/inter-protocol/test/vaultFactory/test-vaultLiquidation.js index c5f2acf5478..12ffdfa38ad 100644 --- a/packages/inter-protocol/test/vaultFactory/test-vaultLiquidation.js +++ b/packages/inter-protocol/test/vaultFactory/test-vaultLiquidation.js @@ -1855,7 +1855,7 @@ test('reinstate vault', async t => { totalDebt: { value: 158n }, totalCollateral: { value: 44n }, totalProceedsReceived: { value: 34n }, - totalShortfallReceived: { value: 224n }, + totalShortfallReceived: { value: 66n }, totalCollateralSold: { value: 8n }, numLiquidatingVaults: 0, numLiquidationsCompleted: 1, From aa76acff026299b5de087a0dd207bff93896908c Mon Sep 17 00:00:00 2001 From: Turadg Aleahmad Date: Mon, 15 May 2023 22:20:00 -0700 Subject: [PATCH 2/2] refactor(vaultManager): inline recordShortfallAndProceeds --- .../src/vaultFactory/vaultManager.js | 69 +++++++++++-------- 1 file changed, 39 insertions(+), 30 deletions(-) diff --git a/packages/inter-protocol/src/vaultFactory/vaultManager.js b/packages/inter-protocol/src/vaultFactory/vaultManager.js index 7d2a56261a6..5d29b085e7c 100644 --- a/packages/inter-protocol/src/vaultFactory/vaultManager.js +++ b/packages/inter-protocol/src/vaultFactory/vaultManager.js @@ -496,28 +496,6 @@ export const prepareVaultManagerKit = ( proceeds, ); }, - - /** @type {(accounting: { overage: Amount<'nat'>, shortfall: Amount<'nat'> }) => void} */ - recordShortfallAndProceeds(accounting) { - trace('recordShortfallAndProceeds', accounting); - const { state } = this; - - const { overage, shortfall } = accounting; - // cumulative values - state.totalOverageReceived = AmountMath.add( - state.totalOverageReceived, - overage, - ); - state.totalShortfallReceived = AmountMath.add( - state.totalShortfallReceived, - shortfall, - ); - state.totalDebt = AmountMath.subtract(state.totalDebt, shortfall); - - void E.when(factoryPowers.getShortfallReporter(), reporter => - E(reporter).increaseLiquidationShortfall(shortfall), - ); - }, sendToReserve(penalty, seat, seatKeyword = 'Collateral') { const invitation = E(reservePublicFacet).makeAddCollateralInvitation(); @@ -555,9 +533,17 @@ export const prepareVaultManagerKit = ( state.liquidatingDebt = AmountMath.add(state.liquidatingDebt, debt); }, - markDoneLiquidating(debt, collateral) { + /** + * + * @param {Amount<'nat'>} debt + * @param {Amount<'nat'>} collateral + * @param {{ overage: Amount<'nat'>, shortfall: Amount<'nat'> }} accounting + */ + markDoneLiquidating(debt, collateral, accounting) { const { state } = this; + // update liquidation state + state.liquidatingCollateral = AmountMath.subtract( state.liquidatingCollateral, collateral, @@ -566,6 +552,24 @@ export const prepareVaultManagerKit = ( state.liquidatingDebt, debt, ); + + // record shortfall and proceeds + + const { overage, shortfall } = accounting; + // cumulative values + state.totalOverageReceived = AmountMath.add( + state.totalOverageReceived, + overage, + ); + state.totalShortfallReceived = AmountMath.add( + state.totalShortfallReceived, + shortfall, + ); + state.totalDebt = AmountMath.subtract(state.totalDebt, shortfall); + + void E.when(factoryPowers.getShortfallReporter(), reporter => + E(reporter).increaseLiquidationShortfall(shortfall), + ); }, /** * If interest was charged between liquidating and liquidated, erase it. @@ -715,8 +719,11 @@ export const prepareVaultManagerKit = ( vaultData.getSize(), ); - facets.helper.recordShortfallAndProceeds(accounting); - facets.helper.markDoneLiquidating(totalDebt, totalCollateral); + facets.helper.markDoneLiquidating( + totalDebt, + totalCollateral, + accounting, + ); } else if (AmountMath.isEmpty(collateralProceeds)) { // Flow #2a @@ -783,8 +790,11 @@ export const prepareVaultManagerKit = ( vaultData.getSize(), ); - facets.helper.recordShortfallAndProceeds(accounting); - facets.helper.markDoneLiquidating(totalDebt, totalCollateral); + facets.helper.markDoneLiquidating( + totalDebt, + totalCollateral, + accounting, + ); } else { // Flow #2b: There's unsold collateral; some vaults may be revived. @@ -875,9 +885,8 @@ export const prepareVaultManagerKit = ( ); facets.helper.sendToReserve(collatRemaining, liqSeat); - facets.helper.markDoneLiquidating(totalDebt, totalCollateral); - facets.helper.recordShortfallAndProceeds({ - overage: accounting.overage, + facets.helper.markDoneLiquidating(totalDebt, totalCollateral, { + ...accounting, shortfall: shortfallToReserve, }); }