Signed-off-by: microwavedcola1 <microwavedcola@gmail.com>
This commit is contained in:
microwavedcola1 2024-07-24 11:54:05 +02:00
parent 1c7d1722f7
commit 526c2bb405
2 changed files with 358 additions and 302 deletions

View File

@ -1,3 +1,5 @@
import { SequenceType } from '@blockworks-foundation/mangolana/lib/globalTypes';
import { sendSignAndConfirmTransactions } from '@blockworks-foundation/mangolana/lib/transactions';
import {
getGovernanceProgramVersion,
getInstructionDataFromBase64,
@ -12,18 +14,11 @@ import {
withInsertTransaction,
withSignOffProposal,
} from '@solana/spl-governance';
import {
Connection,
PublicKey,
Transaction,
TransactionInstruction,
} from '@solana/web3.js';
import { Connection, PublicKey, TransactionInstruction } from '@solana/web3.js';
import { chunk } from 'lodash';
import { createComputeBudgetIx } from '../../src/utils/rpc';
import { updateVoterWeightRecord } from './updateVoteWeightRecord';
import { VsrClient } from './voteStakeRegistryClient';
import { createComputeBudgetIx } from '../../src/utils/rpc';
import { sendSignAndConfirmTransactions } from '@blockworks-foundation/mangolana/lib/transactions';
import { SequenceType } from '@blockworks-foundation/mangolana/lib/globalTypes';
export const MANGO_MINT = 'MangoCzJ36AjZyKwVj3VnYU4GTonjfVEnJmvvWaxLac';
export const MANGO_REALM_PK = new PublicKey(
@ -142,8 +137,32 @@ export const createProposal = async (
);
}
const txChunks = chunk([...instructions, ...insertInstructions], 2);
let txChunks = chunk([...instructions], 2);
await sendSignAndConfirmTransactions({
connection,
wallet,
transactionInstructions: txChunks.map((txChunk) => ({
instructionsSet: [
{
signers: [],
transactionInstruction: createComputeBudgetIx(80000),
},
...txChunk.map((tx) => ({
signers: [],
transactionInstruction: tx,
})),
],
sequenceType: SequenceType.Sequential,
})),
config: {
maxRetries: 5,
autoRetry: true,
maxTxesInBatch: 20,
logFlowInfo: true,
},
});
txChunks = chunk([...insertInstructions], 1);
await sendSignAndConfirmTransactions({
connection,
wallet,

View File

@ -1,7 +1,6 @@
import {
LISTING_PRESETS,
MidPriceImpact,
getMidPriceImpacts,
} from '@blockworks-foundation/mango-v4-settings/lib/helpers/listingTools';
import { AnchorProvider, Wallet } from '@coral-xyz/anchor';
import { BN } from '@project-serum/anchor';
@ -24,18 +23,13 @@ import fs from 'fs';
import { Bank } from '../src/accounts/bank';
import { Group } from '../src/accounts/group';
import { MangoAccount } from '../src/accounts/mangoAccount';
import { OracleProvider } from '../src/accounts/oracle';
import { MangoClient } from '../src/client';
import { NullTokenEditParams } from '../src/clientIxParamBuilder';
import { MANGO_V4_MAIN_GROUP as MANGO_V4_PRIMARY_GROUP } from '../src/constants';
import {
LiqorPriceImpact,
buildGroupGrid,
findLargestAssetBatchUi,
getEquityForMangoAccounts,
} from '../src/risk';
import { getEquityForMangoAccounts } from '../src/risk';
import {
buildFetch,
toNative,
toNativeI80F48ForQuote,
toUiDecimalsForQuote,
} from '../src/utils';
@ -44,6 +38,7 @@ import {
MANGO_GOVERNANCE_PROGRAM,
MANGO_MINT,
MANGO_REALM_PK,
PYTH_SPONSORED_ORACLES,
} from './governanceInstructions/constants';
import { createProposal } from './governanceInstructions/createProposal';
import {
@ -168,27 +163,27 @@ async function updateTokenParams(): Promise<void> {
const instructions: TransactionInstruction[] = [];
const allMangoAccounts = await client.getAllMangoAccounts(group, true);
// const allMangoAccounts = await client.getAllMangoAccounts(group, true);
const stepSize = 1;
// const stepSize = 1;
const ttlLiqorEquityUi = await getTotalLiqorEquity(
client,
group,
allMangoAccounts,
);
// const ttlLiqorEquityUi = await getTotalLiqorEquity(
// client,
// group,
// allMangoAccounts,
// );
const midPriceImpacts = getMidPriceImpacts(group.pis);
// const midPriceImpacts = getMidPriceImpacts(group.pis);
const pisForLiqor: LiqorPriceImpact[][] = [];
// eslint-disable-next-line no-constant-condition
if (false) {
const pisForLiqor: LiqorPriceImpact[][] = await buildGroupGrid(
group,
allMangoAccounts,
stepSize,
);
}
// const pisForLiqor: LiqorPriceImpact[][] = [];
// // eslint-disable-next-line no-constant-condition
// if (false) {
// const pisForLiqor: LiqorPriceImpact[][] = await buildGroupGrid(
// group,
// allMangoAccounts,
// stepSize,
// );
// }
// eslint-disable-next-line no-constant-condition
if (false) {
@ -206,280 +201,324 @@ async function updateTokenParams(): Promise<void> {
);
}
console.log(Array.from(group.banksMapByTokenIndex.values()).length);
Array.from(group.banksMapByTokenIndex.values())
.map((banks) => banks[0])
.sort((a, b) => a.name.localeCompare(b.name))
.forEach(async (bank) => {
if (bank.oracleProvider != OracleProvider.Pyth) {
console.log(`Skipping ${bank.name}, since not pyth`);
return;
}
if (bank.reduceOnly == 1) {
console.log(`Skipping ${bank.name}, since reduceOnly`);
return;
}
const builder = Builder(NullTokenEditParams);
let change = false;
try {
const tier = Object.values(LISTING_PRESETS).find((x) =>
x.initLiabWeight.toFixed(1) === '1.8'
? x.initLiabWeight.toFixed(1) ===
bank?.initLiabWeight.toNumber().toFixed(1) &&
x.reduceOnly === bank.reduceOnly
: x.initLiabWeight.toFixed(1) ===
bank?.initLiabWeight.toNumber().toFixed(1),
);
// try {
const tier = Object.values(LISTING_PRESETS).find((x) =>
x.initLiabWeight.toFixed(1) === '1.8'
? x.initLiabWeight.toFixed(1) ===
bank?.initLiabWeight.toNumber().toFixed(1) &&
x.reduceOnly === bank.reduceOnly
: x.initLiabWeight.toFixed(1) ===
bank?.initLiabWeight.toNumber().toFixed(1),
);
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (
// bank.uiBorrows() == 0 &&
// bank.reduceOnly == 2 &&
// bank.initAssetWeight.toNumber() == 0 &&
// bank.maintAssetWeight.toNumber() == 0
// ) {
// builder.disableAssetLiquidation(true);
// builder.oracleConfig({
// confFilter: 1000,
// maxStalenessSlots: -1,
// });
// change = true;
// }
// }
const maybePythV2Feed = PYTH_SPONSORED_ORACLES.filter(
(x) =>
x[0].replace('/USD', '') ==
(bank.name.includes('BTC')
? 'BTC'
: bank.name.includes('ETH')
? 'ETH'
: bank.name.toUpperCase()),
);
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (bank.uiBorrows() == 0 && bank.reduceOnly == 1) {
// builder.disableAssetLiquidation(true);
// builder.forceWithdraw(true);
// change = true;
// }
// }
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (!tier) {
// console.log(`${bank.name}, no tier found`);
// } else if (tier.preset_name != 'C') {
// if (tier.preset_name.includes('A')) {
// builder.liquidationFee(bank.liquidationFee.toNumber() * 0.2);
// builder.platformLiquidationFee(
// bank.liquidationFee.toNumber() * 0.8,
// );
// } else if (tier.preset_name.includes('B')) {
// builder.liquidationFee(bank.liquidationFee.toNumber() * 0.4);
// builder.platformLiquidationFee(
// bank.liquidationFee.toNumber() * 0.6,
// );
// }
// change = true;
// }
// }
// eslint-disable-next-line no-constant-condition
// if (true) {
// if (!tier) {
// console.log(`${bank.name}, no tier found`);
// } else {
// console.log(
// `${bank.name.padStart(10)}, ${bank.loanFeeRate
// .mul(I80F48.fromNumber(100))
// .toFixed(2)}, ${bank.loanOriginationFeeRate
// .mul(I80F48.fromNumber(100))
// .toFixed(2)}, ${tier?.preset_name.padStart(5)}, ${(
// tier.loanFeeRate * 100
// ).toFixed(2)}, ${(tier!.loanOriginationFeeRate * 100).toFixed(2)}`,
// );
// builder.loanFeeRate(tier!.loanFeeRate);
// builder.loanOriginationFeeRate(tier!.loanOriginationFeeRate);
// builder.flashLoanSwapFeeRate(tier!.loanOriginationFeeRate);
// change = true;
// }
// }
// formulas are sourced from here
// https://www.notion.so/mango-markets/Mango-v4-Risk-parameter-recommendations-d309cdf5faac4aeea7560356e68532ab
// const priceImpact = getPriceImpactForBank(midPriceImpacts, bank);
// const scaleStartQuoteUi = Math.min(
// 50 * ttlLiqorEquityUi,
// 4 * priceImpact.target_amount,
// );
// eslint-disable-next-line no-constant-condition
if (false) {
// Net borrow limits
const netBorrowLimitPerWindowQuote = Math.max(
10_000,
Math.min(bank.uiDeposits() * bank.uiPrice, 300_000) / 3 +
Math.max(0, bank.uiDeposits() * bank.uiPrice - 300_000) / 5,
);
builder.netBorrowLimitPerWindowQuote(
toNativeI80F48ForQuote(netBorrowLimitPerWindowQuote).toNumber(),
);
change = true;
if (
netBorrowLimitPerWindowQuote !=
toUiDecimalsForQuote(bank.netBorrowLimitPerWindowQuote)
) {
console.log(
`${
bank.name
} new - ${netBorrowLimitPerWindowQuote.toLocaleString()}, old - ${toUiDecimalsForQuote(
bank.netBorrowLimitPerWindowQuote,
).toLocaleString()}`,
);
}
}
// Deposit limits
// eslint-disable-next-line no-constant-condition
if (false) {
if (bank.maintAssetWeight.toNumber() > 0) {
{
// Find asset's largest batch in $ we would need to liquidate, batches are extreme points of a range of price drop,
// range is constrained by leverage provided
// i.e. how much volatility we expect
const r = findLargestAssetBatchUi(
pisForLiqor,
bank.name,
Math.round(bank.maintAssetWeight.toNumber() * 100),
100 - Math.round(bank.maintAssetWeight.toNumber() * 100),
stepSize,
);
const maxLiqBatchQuoteUi = r[0];
const maxLiqBatchUi = r[1];
const sellImpact = getPriceImpactForBank(
midPriceImpacts,
bank,
(bank.liquidationFee.toNumber() * 100) / 2,
);
// Deposit limit = sell impact - largest batch
const allowedNewDepositsQuoteUi =
sellImpact.target_amount - maxLiqBatchQuoteUi;
const allowedNewDepositsUi =
sellImpact.target_amount / bank.uiPrice -
maxLiqBatchQuoteUi / bank.uiPrice;
const depositLimitUi = bank.uiDeposits() + allowedNewDepositsUi;
// LOG
// console.log(
// `${bank.name.padStart(20)} ${maxLiqBatchUi
// .toLocaleString()
// .padStart(15)} ${maxLiqBatchQuoteUi
// .toLocaleString()
// .padStart(15)}$ ${sellImpact.target_amount
// .toLocaleString()
// .padStart(12)}$ ${sellImpact.avg_price_impact_percent
// .toLocaleString()
// .padStart(12)}% ${allowedNewDepositsUi
// .toLocaleString()
// .padStart(20)}${allowedNewDepositsQuoteUi
// .toLocaleString()
// .padStart(20)}$ ${bank
// .uiDeposits()
// .toLocaleString()
// .padStart(12)} ${(bank.uiDeposits() * bank.uiPrice)
// .toLocaleString()
// .padStart(12)}$ ${depositLimitUi
// .toLocaleString()
// .padStart(12)}`,
// );
builder.depositLimit(toNative(depositLimitUi, bank.mintDecimals));
change = true;
}
}
}
const params = builder.build();
console.log(
`${bank.name}, ${params.disableAssetLiquidation} ${params.oracleConfig?.maxStalenessSlots} ${params.oracleConfig?.confFilter}`,
);
let ix = await client.program.methods
.tokenEdit(
params.oracle,
params.oracleConfig,
params.groupInsuranceFund,
params.interestRateParams,
params.loanFeeRate,
params.loanOriginationFeeRate,
params.maintAssetWeight,
params.initAssetWeight,
params.maintLiabWeight,
params.initLiabWeight,
params.liquidationFee,
params.stablePriceDelayIntervalSeconds,
params.stablePriceDelayGrowthLimit,
params.stablePriceGrowthLimit,
params.minVaultToDepositsRatio,
params.netBorrowLimitPerWindowQuote !== null
? new BN(params.netBorrowLimitPerWindowQuote)
: null,
params.netBorrowLimitWindowSizeTs !== null
? new BN(params.netBorrowLimitWindowSizeTs)
: null,
params.borrowWeightScaleStartQuote,
params.depositWeightScaleStartQuote,
params.resetStablePrice ?? false,
params.resetNetBorrowLimit ?? false,
params.reduceOnly,
params.name,
params.forceClose,
params.tokenConditionalSwapTakerFeeRate,
params.tokenConditionalSwapMakerFeeRate,
params.flashLoanSwapFeeRate,
params.interestCurveScaling,
params.interestTargetUtilization,
params.maintWeightShiftStart,
params.maintWeightShiftEnd,
params.maintWeightShiftAssetTarget,
params.maintWeightShiftLiabTarget,
params.maintWeightShiftAbort ?? false,
false, // setFallbackOracle, unused
params.depositLimit,
params.zeroUtilRate,
params.platformLiquidationFee,
params.disableAssetLiquidation,
params.collateralFeePerDay,
params.forceWithdraw,
)
.accounts({
group: group.publicKey,
oracle: bank.oracle,
admin: group.admin,
mintInfo: group.mintInfosMapByTokenIndex.get(bank.tokenIndex)
?.publicKey,
fallbackOracle: PublicKey.default,
})
.remainingAccounts([
{
pubkey: bank.publicKey,
isWritable: true,
isSigner: false,
} as AccountMeta,
])
.instruction();
ix = wrapWithForwarder(
ix,
new PublicKey('8SSLjXBEVk9nesbhi9UMCA32uijbVBUqWoKPPQPTekzt'),
new BN((new Date().getTime() / 1000) * 60 * 60 * 24 * 2 * 7), // 2 weeks
);
const tx = new Transaction({ feePayer: wallet.publicKey }).add(ix);
const simulated = await client.connection.simulateTransaction(tx);
if (simulated.value.err) {
console.log('error', simulated.value.logs);
throw simulated.value.logs;
}
if (change) {
instructions.push(ix);
}
} catch (error) {
console.log(`....Skipping ${bank.name}, ${error}`);
if (maybePythV2Feed.length > 0) {
console.log(` - ${bank.name} ${bank.oracle} ${maybePythV2Feed[0][0]}`);
builder.oracle(new PublicKey(maybePythV2Feed[0][1]));
change = true;
} else {
console.log(`Skipping ${bank.name}, cant find pyth feed`);
}
if (
bank.reduceOnly != 1 &&
maybePythV2Feed.length == 0 &&
bank.oracleProvider == OracleProvider.Pyth &&
!['CHAI', 'DAI', 'BLZE', 'MNGO', 'RENDER'].some(
(item) => item == bank.name,
)
) {
throw new Error(`No pyth feed for ${bank.name}`);
}
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (
// bank.uiBorrows() == 0 &&
// bank.reduceOnly == 2 &&
// bank.initAssetWeight.toNumber() == 0 &&
// bank.maintAssetWeight.toNumber() == 0
// ) {
// builder.disableAssetLiquidation(true);
// builder.oracleConfig({
// confFilter: 1000,
// maxStalenessSlots: -1,
// });
// change = true;
// }
// }
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (bank.uiBorrows() == 0 && bank.reduceOnly == 1) {
// builder.disableAssetLiquidation(true);
// builder.forceWithdraw(true);
// change = true;
// }
// }
// // eslint-disable-next-line no-constant-condition
// if (true) {
// if (!tier) {
// console.log(`${bank.name}, no tier found`);
// } else if (tier.preset_name != 'C') {
// if (tier.preset_name.includes('A')) {
// builder.liquidationFee(bank.liquidationFee.toNumber() * 0.2);
// builder.platformLiquidationFee(
// bank.liquidationFee.toNumber() * 0.8,
// );
// } else if (tier.preset_name.includes('B')) {
// builder.liquidationFee(bank.liquidationFee.toNumber() * 0.4);
// builder.platformLiquidationFee(
// bank.liquidationFee.toNumber() * 0.6,
// );
// }
// change = true;
// }
// }
// eslint-disable-next-line no-constant-condition
// if (true) {
// if (!tier) {
// console.log(`${bank.name}, no tier found`);
// } else {
// console.log(
// `${bank.name.padStart(10)}, ${bank.loanFeeRate
// .mul(I80F48.fromNumber(100))
// .toFixed(2)}, ${bank.loanOriginationFeeRate
// .mul(I80F48.fromNumber(100))
// .toFixed(2)}, ${tier?.preset_name.padStart(5)}, ${(
// tier.loanFeeRate * 100
// ).toFixed(2)}, ${(tier!.loanOriginationFeeRate * 100).toFixed(2)}`,
// );
// builder.loanFeeRate(tier!.loanFeeRate);
// builder.loanOriginationFeeRate(tier!.loanOriginationFeeRate);
// builder.flashLoanSwapFeeRate(tier!.loanOriginationFeeRate);
// change = true;
// }
// }
// formulas are sourced from here
// https://www.notion.so/mango-markets/Mango-v4-Risk-parameter-recommendations-d309cdf5faac4aeea7560356e68532ab
// const priceImpact = getPriceImpactForBank(midPriceImpacts, bank);
// const scaleStartQuoteUi = Math.min(
// 50 * ttlLiqorEquityUi,
// 4 * priceImpact.target_amount,
// );
// eslint-disable-next-line no-constant-condition
if (false) {
// Net borrow limits
const netBorrowLimitPerWindowQuote = Math.max(
10_000,
Math.min(bank.uiDeposits() * bank.uiPrice, 300_000) / 3 +
Math.max(0, bank.uiDeposits() * bank.uiPrice - 300_000) / 5,
);
builder.netBorrowLimitPerWindowQuote(
toNativeI80F48ForQuote(netBorrowLimitPerWindowQuote).toNumber(),
);
change = true;
if (
netBorrowLimitPerWindowQuote !=
toUiDecimalsForQuote(bank.netBorrowLimitPerWindowQuote)
) {
console.log(
`${
bank.name
} new - ${netBorrowLimitPerWindowQuote.toLocaleString()}, old - ${toUiDecimalsForQuote(
bank.netBorrowLimitPerWindowQuote,
).toLocaleString()}`,
);
}
}
// // Deposit limits
// // eslint-disable-next-line no-constant-condition
// if (false) {
// if (bank.maintAssetWeight.toNumber() > 0) {
// {
// // Find asset's largest batch in $ we would need to liquidate, batches are extreme points of a range of price drop,
// // range is constrained by leverage provided
// // i.e. how much volatility we expect
// const r = findLargestAssetBatchUi(
// pisForLiqor,
// bank.name,
// Math.round(bank.maintAssetWeight.toNumber() * 100),
// 100 - Math.round(bank.maintAssetWeight.toNumber() * 100),
// stepSize,
// );
// const maxLiqBatchQuoteUi = r[0];
// const maxLiqBatchUi = r[1];
// const sellImpact = getPriceImpactForBank(
// midPriceImpacts,
// bank,
// (bank.liquidationFee.toNumber() * 100) / 2,
// );
// // Deposit limit = sell impact - largest batch
// const allowedNewDepositsQuoteUi =
// sellImpact.target_amount - maxLiqBatchQuoteUi;
// const allowedNewDepositsUi =
// sellImpact.target_amount / bank.uiPrice -
// maxLiqBatchQuoteUi / bank.uiPrice;
// const depositLimitUi = bank.uiDeposits() + allowedNewDepositsUi;
// // LOG
// // console.log(
// // `${bank.name.padStart(20)} ${maxLiqBatchUi
// // .toLocaleString()
// // .padStart(15)} ${maxLiqBatchQuoteUi
// // .toLocaleString()
// // .padStart(15)}$ ${sellImpact.target_amount
// // .toLocaleString()
// // .padStart(12)}$ ${sellImpact.avg_price_impact_percent
// // .toLocaleString()
// // .padStart(12)}% ${allowedNewDepositsUi
// // .toLocaleString()
// // .padStart(20)}${allowedNewDepositsQuoteUi
// // .toLocaleString()
// // .padStart(20)}$ ${bank
// // .uiDeposits()
// // .toLocaleString()
// // .padStart(12)} ${(bank.uiDeposits() * bank.uiPrice)
// // .toLocaleString()
// // .padStart(12)}$ ${depositLimitUi
// // .toLocaleString()
// // .padStart(12)}`,
// // );
// builder.depositLimit(toNative(depositLimitUi, bank.mintDecimals));
// change = true;
// }
// }
// }
const params = builder.build();
if (change) {
// console.log(
// `${bank.name}, ${params.disableAssetLiquidation} ${params.oracleConfig?.maxStalenessSlots} ${params.oracleConfig?.confFilter}`,
// );
// console.log(`${bank.name}, ${bank.oracle} ${params.oracle}`);
}
let ix = await client.program.methods
.tokenEdit(
params.oracle,
params.oracleConfig,
params.groupInsuranceFund,
params.interestRateParams,
params.loanFeeRate,
params.loanOriginationFeeRate,
params.maintAssetWeight,
params.initAssetWeight,
params.maintLiabWeight,
params.initLiabWeight,
params.liquidationFee,
params.stablePriceDelayIntervalSeconds,
params.stablePriceDelayGrowthLimit,
params.stablePriceGrowthLimit,
params.minVaultToDepositsRatio,
params.netBorrowLimitPerWindowQuote !== null
? new BN(params.netBorrowLimitPerWindowQuote)
: null,
params.netBorrowLimitWindowSizeTs !== null
? new BN(params.netBorrowLimitWindowSizeTs)
: null,
params.borrowWeightScaleStartQuote,
params.depositWeightScaleStartQuote,
params.resetStablePrice ?? false,
params.resetNetBorrowLimit ?? false,
params.reduceOnly,
params.name,
params.forceClose,
params.tokenConditionalSwapTakerFeeRate,
params.tokenConditionalSwapMakerFeeRate,
params.flashLoanSwapFeeRate,
params.interestCurveScaling,
params.interestTargetUtilization,
params.maintWeightShiftStart,
params.maintWeightShiftEnd,
params.maintWeightShiftAssetTarget,
params.maintWeightShiftLiabTarget,
params.maintWeightShiftAbort ?? false,
false, // setFallbackOracle, unused
params.depositLimit,
params.zeroUtilRate,
params.platformLiquidationFee,
params.disableAssetLiquidation,
params.collateralFeePerDay,
params.forceWithdraw,
)
.accounts({
group: group.publicKey,
oracle: bank.oracle,
admin: group.admin,
mintInfo: group.mintInfosMapByTokenIndex.get(bank.tokenIndex)
?.publicKey,
fallbackOracle: PublicKey.default,
})
.remainingAccounts([
{
pubkey: bank.publicKey,
isWritable: true,
isSigner: false,
} as AccountMeta,
])
.instruction();
ix = wrapWithForwarder(
ix,
new PublicKey('8SSLjXBEVk9nesbhi9UMCA32uijbVBUqWoKPPQPTekzt'),
new BN(new Date().getTime() / 1000 + 60 * 60 * 24 * 2 * 7), // 2 weeks
);
const tx = new Transaction({ feePayer: wallet.publicKey }).add(ix);
const simulated = await client.connection.simulateTransaction(tx);
if (simulated.value.err) {
console.log('error', simulated.value.logs);
throw simulated.value.logs;
}
if (change) {
instructions.push(ix);
}
// } catch (error) {
// // console.log(error.stack);
// console.log(`....Skipping ${bank.name}, ${error}`);
// }
});
const tokenOwnerRecordPk = await getTokenOwnerRecordAddress(
@ -508,9 +547,7 @@ async function updateTokenParams(): Promise<void> {
walletSigner,
MANGO_DAO_WALLET_GOVERNANCE,
tokenOwnerRecord,
PROPOSAL_TITLE
? PROPOSAL_TITLE
: 'Disable asset liquidation for C tier tokens in mango-v4, part 2',
PROPOSAL_TITLE ? PROPOSAL_TITLE : 'Switch feeds from Pyth V1 to Pyth V2',
PROPOSAL_LINK ?? '',
Object.values(proposals).length,
instructions,