@kamino-finance/klend-sdk 5.13.7 → 5.13.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (38) hide show
  1. package/dist/classes/manager.d.ts +17 -1
  2. package/dist/classes/manager.d.ts.map +1 -1
  3. package/dist/classes/manager.js +20 -0
  4. package/dist/classes/manager.js.map +1 -1
  5. package/dist/classes/obligation.d.ts +1 -0
  6. package/dist/classes/obligation.d.ts.map +1 -1
  7. package/dist/classes/obligation.js +7 -0
  8. package/dist/classes/obligation.js.map +1 -1
  9. package/dist/classes/vault.d.ts +17 -1
  10. package/dist/classes/vault.d.ts.map +1 -1
  11. package/dist/classes/vault.js +66 -0
  12. package/dist/classes/vault.js.map +1 -1
  13. package/dist/classes/vault_types.d.ts +4 -0
  14. package/dist/classes/vault_types.d.ts.map +1 -1
  15. package/dist/client_kamino_manager.d.ts.map +1 -1
  16. package/dist/client_kamino_manager.js +8 -0
  17. package/dist/client_kamino_manager.js.map +1 -1
  18. package/dist/lending_operations/repay_with_collateral_calcs.d.ts +3 -4
  19. package/dist/lending_operations/repay_with_collateral_calcs.d.ts.map +1 -1
  20. package/dist/lending_operations/repay_with_collateral_calcs.js +36 -34
  21. package/dist/lending_operations/repay_with_collateral_calcs.js.map +1 -1
  22. package/dist/lending_operations/repay_with_collateral_operations.d.ts +3 -2
  23. package/dist/lending_operations/repay_with_collateral_operations.d.ts.map +1 -1
  24. package/dist/lending_operations/repay_with_collateral_operations.js +40 -30
  25. package/dist/lending_operations/repay_with_collateral_operations.js.map +1 -1
  26. package/dist/utils/multisig.d.ts +11 -1
  27. package/dist/utils/multisig.d.ts.map +1 -1
  28. package/dist/utils/multisig.js +8 -5
  29. package/dist/utils/multisig.js.map +1 -1
  30. package/package.json +1 -1
  31. package/src/classes/manager.ts +31 -0
  32. package/src/classes/obligation.ts +10 -0
  33. package/src/classes/vault.ts +91 -0
  34. package/src/classes/vault_types.ts +5 -0
  35. package/src/client_kamino_manager.ts +9 -0
  36. package/src/lending_operations/repay_with_collateral_calcs.ts +56 -39
  37. package/src/lending_operations/repay_with_collateral_operations.ts +84 -33
  38. package/src/utils/multisig.ts +19 -5
@@ -2,7 +2,11 @@ import Decimal from 'decimal.js';
2
2
  import { KaminoMarket, KaminoObligation, KaminoReserve, numberToLamportsDecimal } from '../classes';
3
3
  import { PublicKey } from '@solana/web3.js';
4
4
  import { lamportsToDecimal } from '../classes/utils';
5
- import { MaxWithdrawLtvCheck, getMaxWithdrawLtvCheck } from './repay_with_collateral_operations';
5
+ import {
6
+ MaxWithdrawLtvCheck,
7
+ getMaxCollateralFromRepayAmount,
8
+ getMaxWithdrawLtvCheck,
9
+ } from './repay_with_collateral_operations';
6
10
 
7
11
  export function calcRepayAmountWithSlippage(
8
12
  kaminoMarket: KaminoMarket,
@@ -16,7 +20,7 @@ export function calcRepayAmountWithSlippage(
16
20
  repayAmountLamports: Decimal;
17
21
  flashRepayAmountLamports: Decimal;
18
22
  } {
19
- const irSlippageBpsForDebt = obligation
23
+ const interestRateAccrued = obligation
20
24
  .estimateObligationInterestRate(
21
25
  kaminoMarket,
22
26
  debtReserve,
@@ -24,9 +28,9 @@ export function calcRepayAmountWithSlippage(
24
28
  currentSlot
25
29
  )
26
30
  .toDecimalPlaces(debtReserve.state.liquidity.mintDecimals.toNumber(), Decimal.ROUND_CEIL);
27
- // add 0.1% to irSlippageBpsForDebt because we don't want to estimate slightly less than SC and end up not reapying enough
31
+ // add 0.1% to interestRateAccrued because we don't want to estimate slightly less than SC and end up not repaying enough
28
32
  const repayAmountIrAdjusted = amount
29
- .mul(irSlippageBpsForDebt.mul(new Decimal('1.001')))
33
+ .mul(interestRateAccrued.mul(new Decimal('1.001')))
30
34
  .toDecimalPlaces(debtReserve.state.liquidity.mintDecimals.toNumber(), Decimal.ROUND_CEIL);
31
35
 
32
36
  let repayAmount: Decimal;
@@ -34,7 +38,7 @@ export function calcRepayAmountWithSlippage(
34
38
  if (
35
39
  repayAmountIrAdjusted.greaterThanOrEqualTo(
36
40
  lamportsToDecimal(
37
- obligation.borrows.get(debtReserve.address)?.amount || new Decimal(0),
41
+ obligation.getBorrowByReserve(debtReserve.address)?.amount || new Decimal(0),
38
42
  debtReserve.stats.decimals
39
43
  )
40
44
  )
@@ -83,8 +87,7 @@ export function calcMaxWithdrawCollateral(
83
87
  debtReserveAddr: PublicKey,
84
88
  repayAmountLamports: Decimal
85
89
  ): {
86
- repayAmountLamports: Decimal;
87
- withdrawableCollLamports: Decimal;
90
+ maxWithdrawableCollLamports: Decimal;
88
91
  canWithdrawAllColl: boolean;
89
92
  repayingAllDebt: boolean;
90
93
  } {
@@ -92,8 +95,9 @@ export function calcMaxWithdrawCollateral(
92
95
  const borrow = obligation.getBorrowByReserve(debtReserveAddr)!;
93
96
  const depositReserve = market.getReserveByAddress(deposit.reserveAddress)!;
94
97
  const debtReserve = market.getReserveByAddress(borrow.reserveAddress)!;
95
- const depositTotalLamports = deposit.amount.floor();
98
+ const depositTotalLamports = deposit.amount.floor(); // TODO: can remove floor, we have lamports only for deposits
96
99
 
100
+ // Calculate the market value of the remaining debt after repaying
97
101
  const remainingBorrowLamports = borrow.amount.sub(repayAmountLamports).ceil();
98
102
  const remainingBorrowAmount = remainingBorrowLamports.div(debtReserve.getMintFactor());
99
103
  let remainingBorrowsValue = remainingBorrowAmount.mul(debtReserve.getOracleMarketPrice());
@@ -103,17 +107,31 @@ export function calcMaxWithdrawCollateral(
103
107
  .filter((p) => !p.reserveAddress.equals(borrow.reserveAddress))
104
108
  .reduce((acc, b) => acc.add(b.marketValueRefreshed), new Decimal('0'));
105
109
  }
106
- const maxWithdrawLtvCheck = getMaxWithdrawLtvCheck(obligation);
107
110
 
108
- let remainingDepositsValueWithLtv = new Decimal('0');
111
+ const hypotheticalWithdrawLamports = getMaxCollateralFromRepayAmount(
112
+ repayAmountLamports.div(debtReserve.getMintFactor()),
113
+ debtReserve,
114
+ depositReserve
115
+ );
116
+
117
+ // Calculate the max withdraw ltv we can withdraw up to
118
+ const maxWithdrawLtvCheck = getMaxWithdrawLtvCheck(
119
+ obligation,
120
+ repayAmountLamports,
121
+ debtReserve,
122
+ hypotheticalWithdrawLamports,
123
+ depositReserve
124
+ );
125
+ // Calculate the max borrowable value remaining against deposits
126
+ let maxBorrowableValueRemainingAgainstDeposits = new Decimal('0');
109
127
  if (obligation.getDeposits().length > 1) {
110
- remainingDepositsValueWithLtv = obligation
128
+ maxBorrowableValueRemainingAgainstDeposits = obligation
111
129
  .getDeposits()
112
130
  .filter((p) => !p.reserveAddress.equals(deposit.reserveAddress))
113
131
  .reduce((acc, d) => {
114
132
  const { maxLtv, liquidationLtv } = obligation.getLtvForReserve(
115
133
  market,
116
- market.getReserveByAddress(d.reserveAddress)!
134
+ market.getExistingReserveByAddress(d.reserveAddress)
117
135
  );
118
136
  const maxWithdrawLtv =
119
137
  maxWithdrawLtvCheck === MaxWithdrawLtvCheck.LIQUIDATION_THRESHOLD ? liquidationLtv : maxLtv;
@@ -121,11 +139,11 @@ export function calcMaxWithdrawCollateral(
121
139
  }, new Decimal('0'));
122
140
  }
123
141
 
124
- // can withdraw all coll
125
- if (remainingDepositsValueWithLtv.gte(remainingBorrowsValue)) {
142
+ // if the remaining borrow value is less than the
143
+ // this means that the user's ltv is less or equal to the max ltv
144
+ if (maxBorrowableValueRemainingAgainstDeposits.gte(remainingBorrowsValue)) {
126
145
  return {
127
- repayAmountLamports: repayAmountLamports,
128
- withdrawableCollLamports: depositTotalLamports,
146
+ maxWithdrawableCollLamports: depositTotalLamports,
129
147
  canWithdrawAllColl: true,
130
148
  repayingAllDebt: repayAmountLamports.gte(borrow.amount),
131
149
  };
@@ -138,16 +156,15 @@ export function calcMaxWithdrawCollateral(
138
156
  maxWithdrawLtvCheck === MaxWithdrawLtvCheck.LIQUIDATION_THRESHOLD ? collLiquidationLtv : collMaxLtv;
139
157
  const numerator = deposit.marketValueRefreshed
140
158
  .mul(maxWithdrawLtv)
141
- .add(remainingDepositsValueWithLtv)
159
+ .add(maxBorrowableValueRemainingAgainstDeposits)
142
160
  .sub(remainingBorrowsValue);
143
161
 
144
162
  const denominator = depositReserve.getOracleMarketPrice().mul(maxWithdrawLtv);
145
163
  const maxCollWithdrawAmount = numerator.div(denominator);
146
- const withdrawableCollLamports = maxCollWithdrawAmount.mul(depositReserve.getMintFactor()).floor();
164
+ const maxWithdrawableCollLamports = maxCollWithdrawAmount.mul(depositReserve.getMintFactor()).floor();
147
165
 
148
166
  return {
149
- repayAmountLamports: repayAmountLamports,
150
- withdrawableCollLamports,
167
+ maxWithdrawableCollLamports,
151
168
  canWithdrawAllColl: false,
152
169
  repayingAllDebt: repayAmountLamports.gte(borrow.amount),
153
170
  };
@@ -158,7 +175,7 @@ export function estimateDebtRepaymentWithColl(props: {
158
175
  collAmount: Decimal; // in decimals
159
176
  priceDebtToColl: Decimal;
160
177
  slippagePct: Decimal;
161
- flashBorrowReserveFlashLoanFeePercentage: Decimal;
178
+ flashLoanFeePct: Decimal;
162
179
  kaminoMarket: KaminoMarket;
163
180
  debtTokenMint: PublicKey;
164
181
  obligation: KaminoObligation;
@@ -168,34 +185,32 @@ export function estimateDebtRepaymentWithColl(props: {
168
185
  collAmount,
169
186
  priceDebtToColl,
170
187
  slippagePct,
171
- flashBorrowReserveFlashLoanFeePercentage,
188
+ flashLoanFeePct,
172
189
  kaminoMarket,
173
190
  debtTokenMint,
174
191
  obligation,
175
192
  currentSlot,
176
193
  } = props;
177
- const slippage = slippagePct.div('100');
178
- const flashLoanFee = flashBorrowReserveFlashLoanFeePercentage.div('100');
179
- const debtReserve = kaminoMarket.getReserveByMint(debtTokenMint);
180
- if (debtReserve === undefined) {
181
- throw new Error('Debt reserve not found');
182
- }
194
+ const slippageMultiplier = new Decimal(1.0).add(slippagePct.div('100'));
195
+ const flashLoanFeeMultiplier = new Decimal(1.0).add(flashLoanFeePct.div('100'));
196
+
197
+ const debtReserve = kaminoMarket.getExistingReserveByMint(debtTokenMint);
183
198
 
184
- const debtAfterSwap = collAmount.div(new Decimal(1.0).add(slippage)).div(priceDebtToColl);
185
- const debtAfterFlashLoanRepay = debtAfterSwap.div(new Decimal(1.0).add(flashLoanFee));
199
+ const debtAfterSwap = collAmount.div(slippageMultiplier).div(priceDebtToColl);
200
+ const debtAfterFlashLoanRepay = debtAfterSwap.div(flashLoanFeeMultiplier);
186
201
 
187
- const irSlippageBpsForDebt = obligation
202
+ const accruedInterestRate = obligation
188
203
  .estimateObligationInterestRate(
189
204
  kaminoMarket,
190
205
  debtReserve,
191
- obligation?.state.borrows.find((borrow) => borrow.borrowReserve?.equals(debtReserve.address))!,
206
+ obligation.getObligationLiquidityByReserve(debtReserve.address),
192
207
  currentSlot
193
208
  )
194
209
  .toDecimalPlaces(debtReserve.state.liquidity.mintDecimals.toNumber(), Decimal.ROUND_CEIL);
195
210
 
196
211
  // Estimate slightly more, by adding 1% to IR in order to avoid the case where UI users can repay the max we allow them
197
212
  const debtIrAdjusted = debtAfterFlashLoanRepay
198
- .div(irSlippageBpsForDebt.mul(new Decimal('1.01')))
213
+ .div(accruedInterestRate.mul(new Decimal('1.01')))
199
214
  .toDecimalPlaces(debtReserve.state.liquidity.mintDecimals.toNumber(), Decimal.ROUND_CEIL);
200
215
 
201
216
  return debtIrAdjusted;
@@ -205,19 +220,21 @@ export function estimateCollNeededForDebtRepayment(props: {
205
220
  debtAmount: Decimal; // in decimals
206
221
  priceDebtToColl: Decimal;
207
222
  slippagePct: Decimal;
208
- flashBorrowReserveFlashLoanFeePercentage: Decimal;
223
+ flashLoanFeePct: Decimal;
209
224
  }): Decimal {
210
225
  const {
211
226
  debtAmount, // in decimals
212
227
  priceDebtToColl,
213
228
  slippagePct,
214
- flashBorrowReserveFlashLoanFeePercentage,
229
+ flashLoanFeePct,
215
230
  } = props;
216
- const slippage = slippagePct.div('100');
217
- const flashLoanFee = flashBorrowReserveFlashLoanFeePercentage.div('100');
231
+ const slippageRatio = slippagePct.div('100');
232
+ const flashLoanFeeRatio = flashLoanFeePct.div('100');
233
+ const slippageMultiplier = new Decimal(1.0).add(slippageRatio);
234
+ const flashLoanFeeMultiplier = new Decimal(1.0).add(flashLoanFeeRatio);
218
235
 
219
- const debtFlashLoanRepay = debtAmount.mul(new Decimal(1.0).add(flashLoanFee));
220
- const collToSwap = debtFlashLoanRepay.mul(new Decimal(1.0).add(slippage)).mul(priceDebtToColl);
236
+ const debtFlashLoanRepay = debtAmount.mul(flashLoanFeeMultiplier);
237
+ const collToSwap = debtFlashLoanRepay.mul(slippageMultiplier).mul(priceDebtToColl);
221
238
 
222
239
  return collToSwap;
223
240
  }
@@ -83,14 +83,8 @@ export async function getRepayWithCollSwapInputs<QuoteResponse>({
83
83
  flashLoanInfo: FlashLoanInfo;
84
84
  initialInputs: RepayWithCollInitialInputs<QuoteResponse>;
85
85
  }> {
86
- const collReserve = kaminoMarket.getReserveByMint(collTokenMint);
87
- const debtReserve = kaminoMarket.getReserveByMint(debtTokenMint);
88
- if (!collReserve) {
89
- throw new Error(`Collateral reserve with mint ${collTokenMint} not found in market ${kaminoMarket.getAddress()}`);
90
- }
91
- if (!debtReserve) {
92
- throw new Error(`Debt reserve with mint ${debtTokenMint} not found in market ${kaminoMarket.getAddress()}`);
93
- }
86
+ const collReserve = kaminoMarket.getExistingReserveByMint(collTokenMint);
87
+ const debtReserve = kaminoMarket.getExistingReserveByMint(debtTokenMint);
94
88
 
95
89
  const {
96
90
  repayAmountLamports,
@@ -110,7 +104,7 @@ export async function getRepayWithCollSwapInputs<QuoteResponse>({
110
104
  `Collateral position not found for ${collReserve.stats.symbol} reserve ${collReserve.address} in obligation ${obligation.obligationAddress}`
111
105
  );
112
106
  }
113
- const { withdrawableCollLamports } = calcMaxWithdrawCollateral(
107
+ const { maxWithdrawableCollLamports } = calcMaxWithdrawCollateral(
114
108
  kaminoMarket,
115
109
  obligation,
116
110
  collReserve.address,
@@ -118,14 +112,8 @@ export async function getRepayWithCollSwapInputs<QuoteResponse>({
118
112
  repayAmountLamports
119
113
  );
120
114
 
121
- // sanity check: we have extra collateral to swap, but we want to ensure we don't quote for way more than needed and get a bad px
122
- const maxCollNeededFromOracle = finalRepayAmount
123
- .mul(debtReserve.getOracleMarketPrice())
124
- .div(collReserve.getOracleMarketPrice())
125
- .mul('1.1')
126
- .mul(collReserve.getMintFactor())
127
- .ceil();
128
- const inputAmountLamports = Decimal.min(withdrawableCollLamports, maxCollNeededFromOracle);
115
+ const maxCollNeededFromOracle = getMaxCollateralFromRepayAmount(finalRepayAmount, debtReserve, collReserve);
116
+ const inputAmountLamports = Decimal.min(maxWithdrawableCollLamports, maxCollNeededFromOracle);
129
117
 
130
118
  // Build the repay & withdraw collateral tx to get the number of accounts
131
119
  const klendIxs: LeverageIxsOutput = await buildRepayWithCollateralIxs(
@@ -153,7 +141,7 @@ export async function getRepayWithCollSwapInputs<QuoteResponse>({
153
141
  inputAmountLamports,
154
142
  inputMint: collTokenMint,
155
143
  outputMint: debtTokenMint,
156
- amountDebtAtaBalance: new Decimal(0), // only used for kTokens
144
+ amountDebtAtaBalance: undefined, // only used for kTokens
157
145
  };
158
146
 
159
147
  const swapQuote = await quoter(swapQuoteInputs, uniqueKlendAccounts);
@@ -171,13 +159,13 @@ export async function getRepayWithCollSwapInputs<QuoteResponse>({
171
159
  minOutAmountLamports: flashRepayAmountLamports,
172
160
  inputMint: collTokenMint,
173
161
  outputMint: debtTokenMint,
174
- amountDebtAtaBalance: new Decimal(0), // only used for kTokens
162
+ amountDebtAtaBalance: undefined, // only used for kTokens
175
163
  },
176
164
  flashLoanInfo: klendIxs.flashLoanInfo,
177
165
  initialInputs: {
178
166
  debtRepayAmountLamports: repayAmountLamports,
179
167
  flashRepayAmountLamports,
180
- maxCollateralWithdrawLamports: withdrawableCollLamports,
168
+ maxCollateralWithdrawLamports: maxWithdrawableCollLamports,
181
169
  swapQuote,
182
170
  currentSlot,
183
171
  klendAccounts: uniqueKlendAccounts,
@@ -223,17 +211,8 @@ export async function getRepayWithCollIxs<QuoteResponse>({
223
211
  const { debtRepayAmountLamports, flashRepayAmountLamports, maxCollateralWithdrawLamports, swapQuote } = initialInputs;
224
212
  const { inputAmountLamports: collSwapInLamports } = swapInputs;
225
213
 
226
- const collReserve = kaminoMarket.getReserveByMint(collTokenMint);
227
-
228
- if (!collReserve) {
229
- throw new Error(`Collateral reserve with mint ${collTokenMint} not found in market ${kaminoMarket.getAddress()}`);
230
- }
231
-
232
- const debtReserve = kaminoMarket.getReserveByMint(debtTokenMint);
233
-
234
- if (!debtReserve) {
235
- throw new Error(`Debt reserve with mint ${debtTokenMint} not found in market ${kaminoMarket.getAddress()}`);
236
- }
214
+ const collReserve = kaminoMarket.getExistingReserveByMint(collTokenMint);
215
+ const debtReserve = kaminoMarket.getExistingReserveByMint(debtTokenMint);
237
216
 
238
217
  // the client should use these values to prevent this input, but the tx may succeed, so we don't want to fail
239
218
  // there is also a chance that the tx will consume debt token from the user's ata which they would not expect
@@ -326,7 +305,13 @@ async function buildRepayWithCollateralIxs(
326
305
 
327
306
  const requestElevationGroup = !isClosingPosition && obligation.state.elevationGroup !== 0;
328
307
 
329
- const maxWithdrawLtvCheck = getMaxWithdrawLtvCheck(obligation);
308
+ const maxWithdrawLtvCheck = getMaxWithdrawLtvCheck(
309
+ obligation,
310
+ debtRepayAmountLamports,
311
+ debtReserve,
312
+ collWithdrawLamports,
313
+ collReserve
314
+ );
330
315
 
331
316
  // 3. Repay using the flash borrowed funds & withdraw collateral to swap and pay the flash loan
332
317
  let repayAndWithdrawAction;
@@ -393,8 +378,74 @@ async function buildRepayWithCollateralIxs(
393
378
  return res;
394
379
  }
395
380
 
396
- export const getMaxWithdrawLtvCheck = (obligation: KaminoObligation) => {
381
+ export const getMaxWithdrawLtvCheck = (
382
+ obligation: KaminoObligation,
383
+ repayAmountLamports: Decimal,
384
+ debtReserve: KaminoReserve,
385
+ collWithdrawAmount: Decimal,
386
+ collReserve: KaminoReserve
387
+ ) => {
388
+ const [finalLtv, finalMaxLtv] = calculatePostOperationLtv(
389
+ obligation,
390
+ repayAmountLamports,
391
+ debtReserve,
392
+ collWithdrawAmount,
393
+ collReserve
394
+ );
395
+
396
+ if (finalLtv.lte(finalMaxLtv)) {
397
+ return MaxWithdrawLtvCheck.MAX_LTV;
398
+ }
399
+
397
400
  return obligation.refreshedStats.userTotalBorrowBorrowFactorAdjusted.gte(obligation.refreshedStats.borrowLimit)
398
401
  ? MaxWithdrawLtvCheck.LIQUIDATION_THRESHOLD
399
402
  : MaxWithdrawLtvCheck.MAX_LTV;
400
403
  };
404
+
405
+ function calculatePostOperationLtv(
406
+ obligation: KaminoObligation,
407
+ repayAmountLamports: Decimal,
408
+ debtReserve: KaminoReserve,
409
+ collWithdrawAmount: Decimal,
410
+ collReserve: KaminoReserve
411
+ ): [Decimal, Decimal] {
412
+ const repayValue = repayAmountLamports
413
+ .div(debtReserve.getMintFactor())
414
+ .mul(debtReserve.getOracleMarketPrice())
415
+ .mul(debtReserve.getBorrowFactor());
416
+ const collWithdrawValue = collWithdrawAmount.div(collReserve.getMintFactor()).mul(collReserve.getOracleMarketPrice());
417
+
418
+ // Calculate new borrow value and deposit value
419
+ const newBorrowBfValue = Decimal.max(
420
+ new Decimal(0),
421
+ obligation.refreshedStats.userTotalBorrowBorrowFactorAdjusted.sub(repayValue)
422
+ );
423
+ const newDepositValue = Decimal.max(
424
+ new Decimal(0),
425
+ obligation.refreshedStats.userTotalDeposit.sub(collWithdrawValue)
426
+ );
427
+
428
+ const newMaxBorrowableValue = Decimal.max(
429
+ new Decimal(0),
430
+ obligation.refreshedStats.borrowLimit.sub(collWithdrawValue.mul(collReserve.stats.loanToValue))
431
+ );
432
+
433
+ const newMaxLtv = newMaxBorrowableValue.div(newDepositValue);
434
+
435
+ // return final ltv and final max ltv
436
+ return [newBorrowBfValue.div(newDepositValue), newMaxLtv];
437
+ }
438
+
439
+ export function getMaxCollateralFromRepayAmount(
440
+ repayAmount: Decimal,
441
+ debtReserve: KaminoReserve,
442
+ collReserve: KaminoReserve
443
+ ) {
444
+ // sanity check: we have extra collateral to swap, but we want to ensure we don't quote for way more than needed and get a bad px
445
+ return repayAmount
446
+ .mul(debtReserve.getOracleMarketPrice())
447
+ .div(collReserve.getOracleMarketPrice())
448
+ .mul('1.1')
449
+ .mul(collReserve.getMintFactor())
450
+ .ceil();
451
+ }
@@ -1,22 +1,25 @@
1
1
  import { PublicKey } from '@solana/web3.js';
2
2
 
3
- const SQUADS_API_BASE_URL = 'https://4fnetmviidiqkjzenwxe66vgoa0soerr.lambda-url.us-east-1.on.aws/';
3
+ const SQUADS_API_BASE_URL = 'https://4fnetmviidiqkjzenwxe66vgoa0soerr.lambda-url.us-east-1.on.aws';
4
4
 
5
5
  export async function walletIsSquadsMultisig(wallet: PublicKey) {
6
- const response = await fetch(`${SQUADS_API_BASE_URL}/is-multisig?wallet=${wallet.toBase58()}`);
6
+ const response = await fetch(`${SQUADS_API_BASE_URL}/isSquad/${wallet.toBase58()}`);
7
7
  const data = await response.json();
8
8
  const squadsResponse = data as SquadsMultisigResponse;
9
9
  return squadsResponse.isSquad;
10
10
  }
11
11
 
12
12
  // todo: find a way to get the admins number and threshold
13
- export async function getSquadsMultisigAdminsAndThreshold(_wallet: PublicKey): Promise<{
13
+ export async function getSquadsMultisigAdminsAndThreshold(wallet: PublicKey): Promise<{
14
14
  adminsNumber: number;
15
15
  threshold: number;
16
16
  }> {
17
+ const response = await fetch(`${SQUADS_API_BASE_URL}/multisig/${wallet.toBase58()}`);
18
+ const data = await response.json();
19
+ const squadsResponse = data as SquadsMultisigAccountResponse;
17
20
  return {
18
- adminsNumber: 1,
19
- threshold: 1,
21
+ adminsNumber: squadsResponse.keys.length,
22
+ threshold: squadsResponse.threshold,
20
23
  };
21
24
  }
22
25
 
@@ -31,3 +34,14 @@ export interface WalletType {
31
34
  walletAdminsNumber: number;
32
35
  walletThreshold: number;
33
36
  }
37
+
38
+ export type SquadsMultisigAccountResponse = {
39
+ allow_external_execute: boolean;
40
+ authority_index: number;
41
+ bump: number;
42
+ create_key: string;
43
+ keys: number[][];
44
+ ms_change_index: number;
45
+ threshold: number;
46
+ transaction_index: number;
47
+ };