2023-02-21 23:36:59 -08:00
|
|
|
import { AnchorProvider, Wallet } from '@coral-xyz/anchor';
|
2022-12-19 00:42:05 -08:00
|
|
|
import { Cluster, Connection, Keypair, PublicKey } from '@solana/web3.js';
|
2022-08-09 22:32:21 -07:00
|
|
|
import fs from 'fs';
|
2023-02-21 23:36:59 -08:00
|
|
|
import { Group } from '../../src/accounts/group';
|
|
|
|
import { HealthCache } from '../../src/accounts/healthCache';
|
|
|
|
import { HealthType, MangoAccount } from '../../src/accounts/mangoAccount';
|
|
|
|
import { PerpMarket } from '../../src/accounts/perp';
|
|
|
|
import { Serum3Market } from '../../src/accounts/serum3';
|
|
|
|
import { MangoClient } from '../../src/client';
|
|
|
|
import { MANGO_V4_ID } from '../../src/constants';
|
|
|
|
import { toUiDecimalsForQuote } from '../../src/utils';
|
2022-08-09 22:32:21 -07:00
|
|
|
|
2022-09-23 02:43:26 -07:00
|
|
|
const CLUSTER_URL =
|
|
|
|
process.env.CLUSTER_URL_OVERRIDE || process.env.MB_CLUSTER_URL;
|
|
|
|
const PAYER_KEYPAIR =
|
|
|
|
process.env.PAYER_KEYPAIR_OVERRIDE || process.env.MB_PAYER_KEYPAIR;
|
|
|
|
const USER_KEYPAIR =
|
|
|
|
process.env.USER_KEYPAIR_OVERRIDE || process.env.MB_PAYER_KEYPAIR;
|
2022-12-08 21:08:23 -08:00
|
|
|
const GROUP_NUM = Number(process.env.GROUP_NUM || 0);
|
2022-12-20 01:32:33 -08:00
|
|
|
const MANGO_ACCOUNT_PK = process.env.MANGO_ACCOUNT_PK;
|
2022-09-23 02:43:26 -07:00
|
|
|
const CLUSTER: Cluster =
|
|
|
|
(process.env.CLUSTER_OVERRIDE as Cluster) || 'mainnet-beta';
|
|
|
|
|
2022-08-23 04:47:08 -07:00
|
|
|
async function debugUser(
|
|
|
|
client: MangoClient,
|
|
|
|
group: Group,
|
|
|
|
mangoAccount: MangoAccount,
|
2022-10-11 00:38:52 -07:00
|
|
|
): Promise<void> {
|
2023-01-18 11:15:04 -08:00
|
|
|
console.log(mangoAccount.toString(group));
|
2022-08-26 01:08:45 -07:00
|
|
|
|
2022-09-29 06:51:09 -07:00
|
|
|
await mangoAccount.reload(client);
|
2022-08-11 08:44:12 -07:00
|
|
|
|
2022-08-09 22:32:21 -07:00
|
|
|
console.log(
|
|
|
|
'mangoAccount.getEquity() ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
toUiDecimalsForQuote(mangoAccount.getEquity(group)!.toNumber()),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
console.log(
|
|
|
|
'mangoAccount.getHealth(HealthType.init) ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
toUiDecimalsForQuote(
|
|
|
|
mangoAccount.getHealth(group, HealthType.init)!.toNumber(),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
console.log(
|
|
|
|
'HealthCache.fromMangoAccount(group,mangoAccount).health(HealthType.init) ' +
|
|
|
|
toUiDecimalsForQuote(
|
|
|
|
HealthCache.fromMangoAccount(group, mangoAccount)
|
|
|
|
.health(HealthType.init)
|
|
|
|
.toNumber(),
|
|
|
|
),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
console.log(
|
|
|
|
'mangoAccount.getHealthRatio(HealthType.init) ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getHealthRatio(group, HealthType.init)!.toNumber(),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
2022-08-19 12:18:16 -07:00
|
|
|
console.log(
|
|
|
|
'mangoAccount.getHealthRatioUi(HealthType.init) ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getHealthRatioUi(group, HealthType.init),
|
2022-08-19 12:18:16 -07:00
|
|
|
);
|
|
|
|
console.log(
|
|
|
|
'mangoAccount.getHealthRatio(HealthType.maint) ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getHealthRatio(group, HealthType.maint)!.toNumber(),
|
2022-08-19 12:18:16 -07:00
|
|
|
);
|
|
|
|
console.log(
|
|
|
|
'mangoAccount.getHealthRatioUi(HealthType.maint) ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getHealthRatioUi(group, HealthType.maint),
|
2022-08-19 12:18:16 -07:00
|
|
|
);
|
2022-08-09 22:32:21 -07:00
|
|
|
console.log(
|
|
|
|
'mangoAccount.getCollateralValue() ' +
|
2022-09-29 06:51:09 -07:00
|
|
|
toUiDecimalsForQuote(mangoAccount.getCollateralValue(group)!.toNumber()),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
console.log(
|
2022-08-18 22:43:32 -07:00
|
|
|
'mangoAccount.getAssetsValue() ' +
|
2022-08-11 08:44:12 -07:00
|
|
|
toUiDecimalsForQuote(
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getAssetsValue(group, HealthType.init)!.toNumber(),
|
2022-08-11 08:44:12 -07:00
|
|
|
),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
console.log(
|
2022-08-18 22:43:32 -07:00
|
|
|
'mangoAccount.getLiabsValue() ' +
|
2022-08-11 08:44:12 -07:00
|
|
|
toUiDecimalsForQuote(
|
2022-09-29 06:51:09 -07:00
|
|
|
mangoAccount.getLiabsValue(group, HealthType.init)!.toNumber(),
|
2022-08-11 08:44:12 -07:00
|
|
|
),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
|
2022-10-11 00:38:52 -07:00
|
|
|
async function getMaxWithdrawWithBorrowForTokenUiWrapper(
|
|
|
|
token,
|
|
|
|
): Promise<void> {
|
2022-08-23 04:47:08 -07:00
|
|
|
console.log(
|
|
|
|
`mangoAccount.getMaxWithdrawWithBorrowForTokenUi(group, ${token}) ` +
|
|
|
|
mangoAccount.getMaxWithdrawWithBorrowForTokenUi(
|
|
|
|
group,
|
2022-09-01 00:48:23 -07:00
|
|
|
group.banksMapByName.get(token)![0].mint,
|
2022-08-23 04:47:08 -07:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
for (const srcToken of Array.from(group.banksMapByName.keys())) {
|
|
|
|
await getMaxWithdrawWithBorrowForTokenUiWrapper(srcToken);
|
|
|
|
}
|
2022-08-09 22:32:21 -07:00
|
|
|
|
2022-10-11 00:38:52 -07:00
|
|
|
function getMaxSourceForTokenSwapWrapper(src, tgt): void {
|
2022-12-14 05:55:37 -08:00
|
|
|
// Turn on for debugging specific pairs
|
2022-12-16 07:33:37 -08:00
|
|
|
// if (src != 'USDC' || tgt != 'MNGO') return;
|
2022-12-14 05:55:37 -08:00
|
|
|
|
2022-12-16 07:33:37 -08:00
|
|
|
let maxSourceUi;
|
|
|
|
try {
|
|
|
|
maxSourceUi = mangoAccount.getMaxSourceUiForTokenSwap(
|
|
|
|
group,
|
|
|
|
group.banksMapByName.get(src)![0].mint,
|
|
|
|
group.banksMapByName.get(tgt)![0].mint,
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
console.log(`Error for ${src}->${tgt}, ` + error.toString());
|
|
|
|
}
|
2022-12-12 05:54:18 -08:00
|
|
|
|
2023-05-10 02:18:46 -07:00
|
|
|
const maxTargetUi =
|
|
|
|
maxSourceUi *
|
2022-12-02 06:48:43 -08:00
|
|
|
(group.banksMapByName.get(src)![0].uiPrice /
|
|
|
|
group.banksMapByName.get(tgt)![0].uiPrice);
|
2022-12-12 05:54:18 -08:00
|
|
|
|
2022-12-02 06:48:43 -08:00
|
|
|
const sim = mangoAccount.simHealthRatioWithTokenPositionUiChanges(group, [
|
|
|
|
{
|
|
|
|
mintPk: group.banksMapByName.get(src)![0].mint,
|
2023-05-10 02:18:46 -07:00
|
|
|
uiTokenAmount: -maxSourceUi,
|
2022-12-02 06:48:43 -08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
mintPk: group.banksMapByName.get(tgt)![0].mint,
|
2023-05-10 02:18:46 -07:00
|
|
|
uiTokenAmount: maxTargetUi,
|
2022-12-02 06:48:43 -08:00
|
|
|
},
|
|
|
|
]);
|
2022-08-11 08:44:12 -07:00
|
|
|
console.log(
|
|
|
|
`getMaxSourceForTokenSwap ${src.padEnd(4)} ${tgt.padEnd(4)} ` +
|
2022-12-02 06:48:43 -08:00
|
|
|
maxSourceUi.toFixed(3).padStart(10) +
|
|
|
|
`, health ratio after (${sim.toFixed(3).padStart(10)})`,
|
2022-08-11 08:44:12 -07:00
|
|
|
);
|
|
|
|
}
|
2022-12-02 06:48:43 -08:00
|
|
|
for (const srcToken of Array.from(group.banksMapByName.keys()).sort()) {
|
|
|
|
for (const tgtToken of Array.from(group.banksMapByName.keys()).sort()) {
|
2022-08-23 04:47:08 -07:00
|
|
|
getMaxSourceForTokenSwapWrapper(srcToken, tgtToken);
|
|
|
|
}
|
|
|
|
}
|
2022-09-23 00:34:08 -07:00
|
|
|
|
2022-10-11 00:38:52 -07:00
|
|
|
function getMaxForPerpWrapper(perpMarket: PerpMarket): void {
|
2022-12-02 06:48:43 -08:00
|
|
|
const maxQuoteUi = mangoAccount.getMaxQuoteForPerpBidUi(
|
|
|
|
group,
|
|
|
|
perpMarket.perpMarketIndex,
|
|
|
|
);
|
|
|
|
const simMaxQuote = mangoAccount.simHealthRatioWithPerpBidUiChanges(
|
|
|
|
group,
|
|
|
|
perpMarket.perpMarketIndex,
|
|
|
|
maxQuoteUi / perpMarket.uiPrice,
|
|
|
|
);
|
|
|
|
const maxBaseUi = mangoAccount.getMaxBaseForPerpAskUi(
|
|
|
|
group,
|
|
|
|
perpMarket.perpMarketIndex,
|
|
|
|
);
|
|
|
|
const simMaxBase = mangoAccount.simHealthRatioWithPerpAskUiChanges(
|
|
|
|
group,
|
|
|
|
perpMarket.perpMarketIndex,
|
|
|
|
maxBaseUi,
|
2022-09-23 02:43:26 -07:00
|
|
|
);
|
|
|
|
console.log(
|
2022-12-02 06:48:43 -08:00
|
|
|
`getMaxPerp ${perpMarket.name.padStart(
|
|
|
|
10,
|
|
|
|
)} getMaxQuoteForPerpBidUi ${maxQuoteUi
|
|
|
|
.toFixed(3)
|
|
|
|
.padStart(10)} health ratio after (${simMaxQuote
|
|
|
|
.toFixed(3)
|
|
|
|
.padStart(10)}), getMaxBaseForPerpAskUi ${maxBaseUi
|
|
|
|
.toFixed(3)
|
|
|
|
.padStart(10)} health ratio after (${simMaxBase
|
|
|
|
.toFixed(3)
|
|
|
|
.padStart(10)})`,
|
2022-09-23 02:43:26 -07:00
|
|
|
);
|
|
|
|
}
|
2022-09-30 04:33:21 -07:00
|
|
|
for (const perpMarket of Array.from(
|
|
|
|
group.perpMarketsMapByMarketIndex.values(),
|
|
|
|
)) {
|
2022-09-23 02:43:26 -07:00
|
|
|
getMaxForPerpWrapper(perpMarket);
|
|
|
|
}
|
|
|
|
|
2022-10-11 00:38:52 -07:00
|
|
|
function getMaxForSerum3Wrapper(serum3Market: Serum3Market): void {
|
2022-09-23 00:34:08 -07:00
|
|
|
console.log(
|
|
|
|
`getMaxQuoteForSerum3BidUi ${serum3Market.name} ` +
|
|
|
|
mangoAccount.getMaxQuoteForSerum3BidUi(
|
|
|
|
group,
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
),
|
|
|
|
);
|
2023-01-19 23:29:36 -08:00
|
|
|
console.log(
|
|
|
|
`- simHealthRatioWithSerum3BidUiChanges ${serum3Market.name} ` +
|
|
|
|
mangoAccount.simHealthRatioWithSerum3BidUiChanges(
|
|
|
|
group,
|
|
|
|
mangoAccount.getMaxQuoteForSerum3BidUi(
|
|
|
|
group,
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
),
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
HealthType.init,
|
|
|
|
),
|
|
|
|
);
|
2022-09-23 00:34:08 -07:00
|
|
|
console.log(
|
|
|
|
`getMaxBaseForSerum3AskUi ${serum3Market.name} ` +
|
|
|
|
mangoAccount.getMaxBaseForSerum3AskUi(
|
|
|
|
group,
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
),
|
|
|
|
);
|
2023-01-19 23:29:36 -08:00
|
|
|
console.log(
|
|
|
|
`- simHealthRatioWithSerum3BidUiChanges ${serum3Market.name} ` +
|
|
|
|
mangoAccount.simHealthRatioWithSerum3AskUiChanges(
|
|
|
|
group,
|
|
|
|
mangoAccount.getMaxBaseForSerum3AskUi(
|
|
|
|
group,
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
),
|
|
|
|
serum3Market.serumMarketExternal,
|
|
|
|
HealthType.init,
|
|
|
|
),
|
|
|
|
);
|
2022-09-23 00:34:08 -07:00
|
|
|
}
|
|
|
|
for (const serum3Market of Array.from(
|
|
|
|
group.serum3MarketsMapByExternal.values(),
|
|
|
|
)) {
|
|
|
|
getMaxForSerum3Wrapper(serum3Market);
|
|
|
|
}
|
2022-08-09 22:32:21 -07:00
|
|
|
}
|
|
|
|
|
2022-10-11 00:38:52 -07:00
|
|
|
async function main(): Promise<void> {
|
2022-08-09 22:32:21 -07:00
|
|
|
const options = AnchorProvider.defaultOptions();
|
2022-09-23 02:43:26 -07:00
|
|
|
const connection = new Connection(CLUSTER_URL!, options);
|
2022-08-09 22:32:21 -07:00
|
|
|
|
|
|
|
const admin = Keypair.fromSecretKey(
|
2022-09-23 02:43:26 -07:00
|
|
|
Buffer.from(JSON.parse(fs.readFileSync(PAYER_KEYPAIR!, 'utf-8'))),
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
console.log(`Admin ${admin.publicKey.toBase58()}`);
|
|
|
|
|
|
|
|
const adminWallet = new Wallet(admin);
|
|
|
|
const adminProvider = new AnchorProvider(connection, adminWallet, options);
|
|
|
|
const client = MangoClient.connect(
|
|
|
|
adminProvider,
|
2022-09-23 02:43:26 -07:00
|
|
|
CLUSTER,
|
|
|
|
MANGO_V4_ID[CLUSTER],
|
2022-11-04 07:35:40 -07:00
|
|
|
{
|
2023-01-12 11:43:37 -08:00
|
|
|
idsSource: 'api',
|
2022-11-04 07:35:40 -07:00
|
|
|
},
|
2022-08-09 22:32:21 -07:00
|
|
|
);
|
|
|
|
|
2022-09-23 02:43:26 -07:00
|
|
|
const group = await client.getGroupForCreator(admin.publicKey, GROUP_NUM);
|
2022-08-09 22:32:21 -07:00
|
|
|
|
2022-09-23 02:43:26 -07:00
|
|
|
for (const keypair of [USER_KEYPAIR!]) {
|
2022-08-09 22:32:21 -07:00
|
|
|
console.log();
|
|
|
|
const user = Keypair.fromSecretKey(
|
|
|
|
Buffer.from(JSON.parse(fs.readFileSync(keypair, 'utf-8'))),
|
|
|
|
);
|
|
|
|
const userWallet = new Wallet(user);
|
|
|
|
console.log(`User ${userWallet.publicKey.toBase58()}`);
|
2022-08-26 01:09:36 -07:00
|
|
|
|
2023-05-13 02:55:08 -07:00
|
|
|
const mangoAccounts = await client.getAllMangoAccounts(group, true);
|
2022-12-15 00:41:03 -08:00
|
|
|
|
2022-08-18 23:41:57 -07:00
|
|
|
for (const mangoAccount of mangoAccounts) {
|
2022-12-14 12:05:05 -08:00
|
|
|
if (
|
2022-12-20 01:32:33 -08:00
|
|
|
!MANGO_ACCOUNT_PK ||
|
|
|
|
mangoAccount.publicKey.equals(new PublicKey(MANGO_ACCOUNT_PK))
|
2022-12-14 12:05:05 -08:00
|
|
|
) {
|
2023-05-13 02:55:08 -07:00
|
|
|
// console.log();
|
|
|
|
console.log(
|
|
|
|
`${mangoAccount.publicKey
|
|
|
|
.toBase58()
|
|
|
|
.padStart(48)}, health ${toUiDecimalsForQuote(
|
|
|
|
mangoAccount.getHealth(group, HealthType.maint),
|
|
|
|
).toFixed(2)}, ${toUiDecimalsForQuote(
|
|
|
|
mangoAccount.getHealth(group, HealthType.init),
|
|
|
|
).toFixed(2)}`,
|
|
|
|
);
|
|
|
|
// await debugUser(client, group, mangoAccount);
|
2022-12-14 12:05:05 -08:00
|
|
|
}
|
2022-08-18 23:41:57 -07:00
|
|
|
}
|
2022-08-09 22:32:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
process.exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
main();
|
|
|
|
} catch (error) {
|
|
|
|
console.log(error);
|
|
|
|
}
|