mango-v4-ui/store/mangoStore.ts

1193 lines
38 KiB
TypeScript

import dayjs from 'dayjs'
import produce from 'immer'
import create from 'zustand'
import { subscribeWithSelector } from 'zustand/middleware'
import { AnchorProvider, BN, Wallet, web3 } from '@coral-xyz/anchor'
import {
ConfirmOptions,
Connection,
Keypair,
LAMPORTS_PER_SOL,
PublicKey,
TransactionInstruction,
} from '@solana/web3.js'
import { OpenOrders, Order } from '@project-serum/serum/lib/market'
import { Orderbook } from '@project-serum/serum'
import { Wallet as WalletAdapter } from '@solana/wallet-adapter-react'
import {
MangoClient,
Group,
MangoAccount,
Serum3Market,
MANGO_V4_ID,
Bank,
PerpOrder,
PerpPosition,
BookSide,
ParsedFillEvent,
getLargestPerpPositions,
getClosestToLiquidationPerpPositions,
} from '@blockworks-foundation/mango-v4'
import EmptyWallet from '../utils/wallet'
import { TransactionNotification, notify } from '../utils/notifications'
import {
getNFTsByOwner,
getTokenAccountsByOwnerWithWrappedSol,
TokenAccount,
} from '../utils/tokens'
import { Token } from '../types/jupiter'
import {
CONNECTION_COMMITMENT,
DEFAULT_MARKET_NAME,
INPUT_TOKEN_DEFAULT,
LAST_ACCOUNT_KEY,
MANGO_DATA_API_URL,
MANGO_MAINNET_GROUP,
OUTPUT_TOKEN_DEFAULT,
PAGINATION_PAGE_LENGTH,
RPC_PROVIDER_KEY,
SWAP_MARGIN_KEY,
} from '../utils/constants'
import {
ActivityFeed,
EmptyObject,
OrderbookL2,
PerpStatsItem,
PerpTradeHistory,
SerumEvent,
SpotBalances,
SpotTradeHistory,
SwapHistoryItem,
TradeForm,
NFT,
TourSettings,
ThemeData,
PositionStat,
OrderbookTooltip,
SwapTypes,
} from 'types'
import spotBalancesUpdater from './spotBalancesUpdater'
import { PerpMarket } from '@blockworks-foundation/mango-v4'
import perpPositionsUpdater from './perpPositionsUpdater'
import {
DEFAULT_PRIORITY_FEE,
LITE_RPC_URL,
TRITON_DEDICATED_URL,
} from '@components/settings/RpcSettings'
import {
IExecutionLineAdapter,
IOrderLineAdapter,
} from '@public/charting_library/charting_library'
import { nftThemeMeta } from 'utils/theme'
import { OrderTypes } from 'utils/tradeForm'
import { usePlausible } from 'next-plausible'
import { collectTxConfirmationData } from 'utils/transactionConfirmationData'
import { TxCallbackOptions } from '@blockworks-foundation/mango-v4/dist/types/src/client'
const ENDPOINTS = [
{
name: 'mainnet-beta',
url: process.env.NEXT_PUBLIC_ENDPOINT || TRITON_DEDICATED_URL,
websocket: process.env.NEXT_PUBLIC_ENDPOINT || TRITON_DEDICATED_URL,
custom: false,
},
{
name: 'devnet',
url: 'https://realms-develope-935c.devnet.rpcpool.com/67f608dc-a353-4191-9c34-293a5061b536',
websocket:
'https://realms-develope-935c.devnet.rpcpool.com/67f608dc-a353-4191-9c34-293a5061b536',
custom: false,
},
]
const options = {
...AnchorProvider.defaultOptions(),
preflightCommitment: 'confirmed',
} as ConfirmOptions
export const CLUSTER: 'mainnet-beta' | 'devnet' = 'mainnet-beta'
const ENDPOINT = ENDPOINTS.find((e) => e.name === CLUSTER) || ENDPOINTS[0]
export const emptyWallet = new EmptyWallet(Keypair.generate())
const initMangoClient = (
provider: AnchorProvider,
opts: {
prioritizationFee: number
prependedGlobalAdditionalInstructions: TransactionInstruction[]
multipleConnections: Connection[]
} = {
prioritizationFee: DEFAULT_PRIORITY_FEE,
prependedGlobalAdditionalInstructions: [],
multipleConnections: [],
},
//for analytics use
telemetry: ReturnType<typeof usePlausible> | null,
): MangoClient => {
return MangoClient.connect(provider, CLUSTER, MANGO_V4_ID[CLUSTER], {
prioritizationFee: opts.prioritizationFee,
fallbackOracleConfig: [
new PublicKey('Dpw1EAVrSB1ibxiDQyTAW6Zip3J4Btk2x4SgApQCeFbX'), // USDC pyth oracle
new PublicKey('5cN76Xm2Dtx9MnrQqBDeZZRsWruTTcw37UruznAdSvvE'), //Bsol
new PublicKey('AxaxyeDT8JnWERSaTKvFXvPKkEdxnamKSqpWbsSjYg1g'), //jitoSol
new PublicKey('5CKzb9j4ChgLUt8Gfm5CNGLN6khXKiqMbnGAW4cgXgxK'), //mSol
new PublicKey('DBE3N8uNjhKPRHfANdwGvCZghWXyLPdqdSbEW2XFwBiX'), //bonk
new PublicKey('p8WhggEpj4bTQJpGqPANiqG2CWUxooxWBWzi5qhrdzy'), //blze
new PublicKey('6B23K3tkb51vLZA14jcEQVCA1pfHptzEHFA93V5dYwbT'), //wif
new PublicKey('GHKcxocPyzSjy7tWApQjKRkDNuVXd4Kk624zhuaR7xhC'), //mnde
new PublicKey('FLroEBBA4Fa8ENqfBmqyypq8U6ai2mD7c5k6Vfb2PWzv'), //MangoSol
new PublicKey('6zBkSKhAqLT2SNRbzTbrom2siKhVZ6SLQcFPnvyexdTE'), //DualSol
new PublicKey('318uRUE2RuYpvv1VwxC4eJwViDrRrxUTTqoUBV1cgUYi'), //HubSol
new PublicKey('91yrNSV8mofYcP6NCsHNi2YgNxwukBenv5MCRFD92Rgp'), //Jsol
new PublicKey('Am5rswhcxQhqviDXuaiZnLvkpmB4iJEdxmhqMMZDV3KJ'), //DigitSol
new PublicKey('9gFehBozPdWafFfPiZRbub2yUmwYJrGMvguKHii7cMTA'), //CompassSol
],
multipleConnections: opts.multipleConnections,
idsSource: 'api',
prependedGlobalAdditionalInstructions:
opts.prependedGlobalAdditionalInstructions,
postSendTxCallback: async (txCallbackOptions: TxCallbackOptions) => {
if (telemetry) {
telemetry('postSendTx', {
props: { fee: opts.prioritizationFee },
})
}
notify({
title: 'Transaction sent',
description: 'Waiting for confirmation',
type: 'confirm',
txid: txCallbackOptions.txid,
})
collectTxConfirmationData(
provider.connection.rpcEndpoint,
opts.prioritizationFee,
txCallbackOptions,
)
},
})
}
const createBackupConnections = (
primaryConnection: Connection,
): Connection[] => {
const liteRpcConnection = new Connection(LITE_RPC_URL)
const backupConnections = [
liteRpcConnection,
new Connection('https://idalina-qy4oxi-fast-mainnet.helius-rpc.com/'),
]
if (primaryConnection.rpcEndpoint !== TRITON_DEDICATED_URL) {
const conn = new Connection(TRITON_DEDICATED_URL)
backupConnections.push(conn)
}
return backupConnections
}
export const DEFAULT_TRADE_FORM: TradeForm = {
side: 'buy',
price: undefined,
baseSize: '',
quoteSize: '',
tradeType: OrderTypes.LIMIT,
triggerPrice: '',
postOnly: false,
ioc: false,
reduceOnly: false,
}
export type AccountPageTab =
| 'overview'
| 'balances'
| 'trade:positions'
| 'trade:orders'
| 'trade:unsettled'
| 'history'
export type MangoStore = {
accountPageTab: AccountPageTab
activityFeed: {
feed: Array<ActivityFeed>
loading: boolean
queryParams: string
}
connected: boolean
connection: Connection
group: Group | undefined
groupLoaded: boolean
client: MangoClient
showUserSetup: boolean
mangoAccount: {
current: MangoAccount | undefined
initialLoad: boolean
lastSlot: number
openOrderAccounts: OpenOrders[]
openOrders: Record<string, Order[] | PerpOrder[]>
perpPositions: PerpPosition[]
spotBalances: SpotBalances
swapHistory: {
data: SwapHistoryItem[]
initialLoad: boolean
loading: boolean
}
tradeHistory: {
data: Array<SpotTradeHistory | PerpTradeHistory>
loading: boolean
}
}
mangoAccounts: MangoAccount[]
markets: Serum3Market[] | undefined
transactionNotificationIdCounter: number
transactionNotifications: Array<TransactionNotification>
perpMarkets: PerpMarket[]
perpStats: {
loading: boolean
data: PerpStatsItem[] | null
positions: {
initialLoad: boolean
loading: boolean
largest: PositionStat[]
closestToLiq: PositionStat[]
}
}
orderbookTooltip: OrderbookTooltip | undefined
prependedGlobalAdditionalInstructions: TransactionInstruction[]
priorityFee: number
selectedMarket: {
name: string | undefined
current: Serum3Market | PerpMarket | undefined
fills: (ParsedFillEvent | SerumEvent)[]
bidsAccount: BookSide | Orderbook | undefined
asksAccount: BookSide | Orderbook | undefined
orderbook: OrderbookL2
markPrice: number
lastSeenSlot: {
bids: number
asks: number
}
}
serumMarkets: Serum3Market[]
serumOrders: Order[] | undefined
settings: {
loading: boolean
tours: TourSettings
uiLocked: boolean
}
successAnimation: {
swap: boolean
theme: boolean
trade: boolean
}
swap: {
inputBank: Bank | undefined
outputBank: Bank | undefined
inputTokenInfo: Token | undefined
outputTokenInfo: Token | undefined
margin: boolean
slippage: number
swapMode: 'ExactIn' | 'ExactOut'
amountIn: string
amountOut: string
flipPrices: boolean
swapOrTrigger: SwapTypes
triggerPrice: string
}
set: (x: (x: MangoStore) => void) => void
themeData: ThemeData
tradeForm: TradeForm
tradingView: {
orderLines: Map<string | BN, IOrderLineAdapter>
tradeExecutions: Map<string, IExecutionLineAdapter>
}
wallet: {
tokens: TokenAccount[]
nfts: {
data: NFT[] | []
initialLoad: boolean
loading: boolean
}
}
window: {
width: number
height: number
}
telemetry: ReturnType<typeof usePlausible> | null
actions: {
fetchActivityFeed: (
mangoAccountPk: string,
offset?: number,
params?: string,
limit?: number,
) => Promise<void>
fetchGroup: () => Promise<void>
reloadMangoAccount: (slot?: number) => Promise<void>
fetchMangoAccounts: (ownerPk: PublicKey) => Promise<void>
fetchNfts: (connection: Connection, walletPk: PublicKey) => void
fetchOpenOrders: (refetchMangoAccount?: boolean) => Promise<void>
fetchPerpStats: () => void
fetchPositionsStats: () => void
fetchSwapHistory: (
mangoAccountPk: string,
timeout?: number,
offset?: number,
limit?: number,
) => Promise<void>
fetchTourSettings: (walletPk: string) => void
fetchWalletTokens: (walletPk: PublicKey) => Promise<void>
connectMangoClientWithWallet: (wallet: WalletAdapter) => Promise<void>
loadMarketFills: () => Promise<void>
updateConnection: (url: string) => void
setPrependedGlobalAdditionalInstructions: (
instructions: TransactionInstruction[],
) => void
updateFee: (
feeMultiplier: number,
fee: number,
telemetry: ReturnType<typeof usePlausible> | null,
) => Promise<void>
}
}
const mangoStore = create<MangoStore>()(
subscribeWithSelector((_set, get) => {
let rpcUrl = ENDPOINT.url
let swapMargin = true
if (typeof window !== 'undefined' && CLUSTER === 'mainnet-beta') {
const urlFromLocalStorage = localStorage.getItem(RPC_PROVIDER_KEY)
const swapMarginFromLocalStorage = localStorage.getItem(SWAP_MARGIN_KEY)
rpcUrl = urlFromLocalStorage
? JSON.parse(urlFromLocalStorage)
: ENDPOINT.url
swapMargin = swapMarginFromLocalStorage
? JSON.parse(swapMarginFromLocalStorage)
: true
}
let connection: Connection
try {
// if connection is using Triton RpcPool then use whirligig
// https://docs.triton.one/project-yellowstone/whirligig-websockets
if (rpcUrl.includes('rpcpool')) {
connection = new web3.Connection(rpcUrl, {
wsEndpoint: `${rpcUrl.replace('http', 'ws')}/whirligig/`,
commitment: CONNECTION_COMMITMENT,
})
} else {
connection = new web3.Connection(rpcUrl, CONNECTION_COMMITMENT)
}
} catch {
connection = new web3.Connection(ENDPOINT.url, CONNECTION_COMMITMENT)
}
const provider = new AnchorProvider(connection, emptyWallet, options)
provider.opts.skipPreflight = true
const backupConnections = createBackupConnections(connection)
const client = initMangoClient(
provider,
{
prioritizationFee: DEFAULT_PRIORITY_FEE,
prependedGlobalAdditionalInstructions: [],
multipleConnections: backupConnections,
},
null,
)
return {
accountPageTab: 'overview',
activityFeed: {
feed: [],
loading: true,
queryParams: '',
},
connected: false,
connection,
group: undefined,
groupLoaded: false,
client,
showUserSetup: false,
mangoAccount: {
current: undefined,
initialLoad: true,
lastSlot: 0,
openOrderAccounts: [],
openOrders: {},
perpPositions: [],
spotBalances: {},
swapHistory: { data: [], loading: true, initialLoad: true },
tradeHistory: { data: [], loading: true },
},
mangoAccounts: [],
markets: undefined,
transactionNotificationIdCounter: 0,
transactionNotifications: [],
perpMarkets: [],
perpStats: {
loading: false,
data: [],
positions: {
initialLoad: true,
loading: true,
largest: [],
closestToLiq: [],
},
},
orderbookTooltip: undefined,
priorityFee: DEFAULT_PRIORITY_FEE,
prependedGlobalAdditionalInstructions: [],
selectedMarket: {
name: 'SOL-PERP',
current: undefined,
fills: [],
bidsAccount: undefined,
asksAccount: undefined,
lastSeenSlot: {
bids: 0,
asks: 0,
},
orderbook: {
bids: [],
asks: [],
},
markPrice: 0,
},
serumMarkets: [],
serumOrders: undefined,
set: (fn) => _set(produce(fn)),
settings: {
loading: false,
tours: {
account_tour_seen: true,
swap_tour_seen: true,
trade_tour_seen: true,
wallet_pk: '',
},
uiLocked: true,
},
successAnimation: {
swap: false,
theme: false,
trade: false,
},
swap: {
inputBank: undefined,
outputBank: undefined,
inputTokenInfo: undefined,
outputTokenInfo: undefined,
margin: swapMargin,
slippage: 0.5,
swapMode: 'ExactIn',
amountIn: '',
amountOut: '',
flipPrices: false,
swapOrTrigger: 'swap',
triggerPrice: '',
},
themeData: nftThemeMeta.default,
tradeForm: DEFAULT_TRADE_FORM,
tradingView: {
orderLines: new Map(),
tradeExecutions: new Map(),
},
wallet: {
tokens: [],
nfts: {
data: [],
loading: false,
initialLoad: true,
},
},
window: {
width: 0,
height: 0,
},
telemetry: null,
actions: {
fetchActivityFeed: async (
mangoAccountPk: string,
offset = 0,
params = '',
limit = PAGINATION_PAGE_LENGTH,
) => {
const set = get().set
const loadedFeed = mangoStore.getState().activityFeed.feed
try {
const response = await fetch(
`${MANGO_DATA_API_URL}/stats/activity-feed?mango-account=${mangoAccountPk}&offset=${offset}&limit=${limit}${
params ? params : ''
}`,
)
const parsedResponse: null | EmptyObject | Array<ActivityFeed> =
await response.json()
if (Array.isArray(parsedResponse)) {
const entries = Object.entries(parsedResponse).sort((a, b) =>
b[0].localeCompare(a[0]),
)
const latestFeed = entries
.map(([key, value]) => {
// ETH should be renamed to ETH (Portal) in the database
const symbol = value.activity_details.symbol
if (symbol === 'ETH') {
value.activity_details.symbol = 'ETH (Portal)'
}
return {
...value,
symbol: key,
}
})
.sort(
(a, b) =>
dayjs(b.block_datetime).unix() -
dayjs(a.block_datetime).unix(),
)
// only add to current feed if data request is offset and the mango account hasn't changed
const combinedFeed =
offset !== 0 ? loadedFeed.concat(latestFeed) : latestFeed
set((state) => {
state.activityFeed.feed = combinedFeed
})
}
} catch (e) {
console.error('Failed to fetch account activity feed', e)
} finally {
set((state) => {
state.activityFeed.loading = false
})
}
},
fetchGroup: async () => {
try {
const set = get().set
const client = get().client
const group = await client.getGroup(MANGO_MAINNET_GROUP)
let selectedMarketName = get().selectedMarket.name
if (!selectedMarketName) {
selectedMarketName = DEFAULT_MARKET_NAME
}
const inputBank =
group?.banksMapByName.get(INPUT_TOKEN_DEFAULT)?.[0]
const outputBank =
group?.banksMapByName.get(OUTPUT_TOKEN_DEFAULT)?.[0]
const serumMarkets = Array.from(
group.serum3MarketsMapByExternal.values(),
).map((m) => {
// remove this when market name is updated on chain via governance
if (m.name === 'MSOL/SOL') {
m.name = 'mSOL/SOL'
}
if (m.name === 'RNDR/USDC') {
m.name = 'RENDER/USDC'
}
return m
})
const perpMarkets = Array.from(group.perpMarketsMapByName.values())
.filter(
(p) =>
p.publicKey.toString() !==
'9Y8paZ5wUpzLFfQuHz8j2RtPrKsDtHx9sbgFmWb5abCw',
)
.sort((a, b) => a.name.localeCompare(b.name))
const selectedMarket =
serumMarkets.find((m) => m.name === selectedMarketName) ||
perpMarkets.find((m) => m.name === selectedMarketName) ||
serumMarkets[0]
set((state) => {
state.group = group
state.groupLoaded = true
state.serumMarkets = serumMarkets
state.perpMarkets = perpMarkets
state.selectedMarket.current = selectedMarket
if (!state.swap.inputBank && !state.swap.outputBank) {
state.swap.inputBank = inputBank
state.swap.outputBank = outputBank
} else {
state.swap.inputBank = group.getFirstBankByMint(
state.swap.inputBank!.mint,
)
state.swap.outputBank = group.getFirstBankByMint(
state.swap.outputBank!.mint,
)
}
})
} catch (e) {
notify({ type: 'info', title: 'Unable to refresh data' })
console.error('Error fetching group', e)
}
},
reloadMangoAccount: async (confirmationSlot) => {
const set = get().set
const actions = get().actions
try {
const group = get().group
const client = get().client
const mangoAccount = get().mangoAccount.current
if (!group) throw new Error('Group not loaded')
if (!mangoAccount)
throw new Error('No mango account exists for reload')
const { value: reloadedMangoAccount, slot } =
await mangoAccount.reloadWithSlot(client)
const lastSlot = get().mangoAccount.lastSlot
if (
!confirmationSlot ||
(confirmationSlot && slot >= confirmationSlot)
) {
if (slot > lastSlot) {
const ma = get().mangoAccounts.find((ma) =>
ma.publicKey.equals(reloadedMangoAccount.publicKey),
)
if (ma) {
Object.assign(ma, reloadedMangoAccount)
}
set((state) => {
state.mangoAccount.current = reloadedMangoAccount
state.mangoAccount.lastSlot = slot
})
}
} else if (confirmationSlot && slot < confirmationSlot) {
actions.reloadMangoAccount(confirmationSlot)
}
} catch (e) {
console.error('Error reloading mango acct', e)
actions.reloadMangoAccount()
} finally {
set((state) => {
state.mangoAccount.initialLoad = false
})
}
},
fetchMangoAccounts: async (ownerPk: PublicKey) => {
const set = get().set
const actions = get().actions
try {
const group = get().group
const client = get().client
const selectedMangoAccount = get().mangoAccount.current
const hasLevCloseInUrl = window.location.href.includes(
'lev-stake-sol-inspect',
)
if (!group) throw new Error('Group not loaded')
if (!client) throw new Error('Client not loaded')
const [ownerMangoAccounts, delegateAccounts] = await Promise.all([
client.getMangoAccountsForOwner(group, ownerPk),
client.getMangoAccountsForDelegate(group, ownerPk),
])
const mangoAccounts = [
...ownerMangoAccounts,
...delegateAccounts,
].filter(
(acc) => hasLevCloseInUrl || !acc.name.includes('Leverage Stake'),
)
const selectedAccountIsNotInAccountsList = mangoAccounts.find(
(x) =>
x.publicKey.toBase58() ===
selectedMangoAccount?.publicKey.toBase58(),
)
if (!mangoAccounts?.length) {
set((state) => {
state.mangoAccounts = []
state.mangoAccount.current = undefined
})
return
}
let newSelectedMangoAccount = selectedMangoAccount
if (!selectedMangoAccount || !selectedAccountIsNotInAccountsList) {
const lastAccount = localStorage.getItem(LAST_ACCOUNT_KEY)
newSelectedMangoAccount = mangoAccounts[0]
let lastViewedAccount
if (typeof lastAccount === 'string') {
try {
lastViewedAccount = mangoAccounts.find(
(m) => m.publicKey.toString() === JSON.parse(lastAccount),
)
} catch (e) {
console.error('Error parsing last account', e)
}
newSelectedMangoAccount = lastViewedAccount || mangoAccounts[0]
}
}
if (newSelectedMangoAccount) {
await newSelectedMangoAccount.reloadSerum3OpenOrders(client)
set((state) => {
state.mangoAccount.current = newSelectedMangoAccount
state.mangoAccount.initialLoad = false
})
actions.fetchOpenOrders()
}
await Promise.all(
mangoAccounts.map((ma) => ma.reloadSerum3OpenOrders(client)),
)
set((state) => {
state.mangoAccounts = mangoAccounts
})
} catch (e) {
console.error('Error fetching mango accts', e)
} finally {
set((state) => {
state.mangoAccount.initialLoad = false
})
}
},
fetchNfts: async (connection: Connection, ownerPk: PublicKey) => {
const set = get().set
set((state) => {
state.wallet.nfts.loading = true
})
try {
const nfts = await getNFTsByOwner(ownerPk, connection)
set((state) => {
state.wallet.nfts.data = nfts
})
} catch (error) {
console.warn('Error: unable to fetch nfts.', error)
} finally {
const notLoaded = mangoStore.getState().wallet.nfts.initialLoad
set((state) => {
state.wallet.nfts.loading = false
if (notLoaded) {
state.wallet.nfts.initialLoad = false
}
})
}
},
fetchOpenOrders: async (refetchMangoAccount = false) => {
const set = get().set
const client = get().client
const group = get().group
if (refetchMangoAccount) {
await get().actions.reloadMangoAccount()
}
const mangoAccount = get().mangoAccount.current
if (!mangoAccount || !group) return
try {
const openOrders: Record<string, Order[] | PerpOrder[]> = {}
let serumOpenOrderAccounts: OpenOrders[] = []
const activeSerumMarketIndices = [
...new Set(mangoAccount.serum3Active().map((s) => s.marketIndex)),
]
if (activeSerumMarketIndices.length) {
await Promise.all(
activeSerumMarketIndices.map(async (serum3Orders) => {
const market =
group.getSerum3MarketByMarketIndex(serum3Orders)
if (market) {
const orders =
await mangoAccount.loadSerum3OpenOrdersForMarket(
client,
group,
market.serumMarketExternal,
)
openOrders[market.serumMarketExternal.toString()] = orders
}
}),
)
}
if (mangoAccount.serum3Active().length) {
serumOpenOrderAccounts = Array.from(
mangoAccount.serum3OosMapByMarketIndex.values(),
)
await mangoAccount.loadSerum3OpenOrdersAccounts(client)
}
const activePerpMarketIndices = [
...new Set(
mangoAccount.perpOrdersActive().map((p) => p.orderMarket),
),
]
await Promise.all(
activePerpMarketIndices.map(async (perpMktIndex) => {
const market = group.getPerpMarketByMarketIndex(perpMktIndex)
const orders = await mangoAccount.loadPerpOpenOrdersForMarket(
client,
group,
perpMktIndex,
market._bids ? false : true,
)
openOrders[market.publicKey.toString()] = orders
}),
)
set((s) => {
s.mangoAccount.openOrders = openOrders
s.mangoAccount.openOrderAccounts = serumOpenOrderAccounts
})
} catch (e) {
console.error('Failed loading open orders ', e)
}
},
fetchPerpStats: async () => {
const set = get().set
const group = get().group
if (!group) return []
set((state) => {
state.perpStats.loading = true
})
try {
const response = await fetch(
`${MANGO_DATA_API_URL}/perp-historical-stats?mango-group=${group?.publicKey.toString()}`,
)
const data = await response.json()
set((state) => {
state.perpStats.data = data
state.perpStats.loading = false
})
} catch (error) {
set((state) => {
state.perpStats.loading = false
})
console.log('Failed to fetch perp stats data', error)
notify({
title: 'Failed to fetch perp stats data',
type: 'error',
})
}
},
fetchPositionsStats: async () => {
const set = get().set
const group = get().group
const client = get().client
if (!group) return
try {
const allMangoAccounts = await client.getAllMangoAccounts(
group,
true,
)
if (allMangoAccounts && allMangoAccounts.length) {
const [largestPositions, closestToLiq]: [
PositionStat[],
PositionStat[],
] = await Promise.all([
getLargestPerpPositions(client, group, allMangoAccounts),
getClosestToLiquidationPerpPositions(
client,
group,
allMangoAccounts,
),
])
set((state) => {
if (largestPositions && largestPositions.length) {
const positionsToShow = largestPositions.slice(0, 5)
for (const position of positionsToShow) {
const ma = allMangoAccounts.find(
(acc) => acc.publicKey === position.mangoAccount,
)
position.account = ma
}
state.perpStats.positions.largest = positionsToShow
}
if (closestToLiq && closestToLiq.length) {
const positionsToShow = closestToLiq.slice(0, 5)
for (const position of positionsToShow) {
const ma = allMangoAccounts.find(
(acc) => acc.publicKey === position.mangoAccount,
)
position.account = ma
}
state.perpStats.positions.closestToLiq = positionsToShow
}
})
}
} catch (e) {
console.log('failed to fetch perp positions stats', e)
} finally {
const notLoaded =
mangoStore.getState().perpStats.positions.initialLoad
set((state) => {
state.perpStats.positions.loading = false
if (notLoaded) {
state.perpStats.positions.initialLoad = false
}
})
}
},
fetchSwapHistory: async (
mangoAccountPk: string,
timeout = 0,
offset = 0,
limit = PAGINATION_PAGE_LENGTH,
) => {
const set = get().set
const loadedSwapHistory =
mangoStore.getState().mangoAccount.swapHistory.data
setTimeout(async () => {
try {
const history = await fetch(
`${MANGO_DATA_API_URL}/stats/swap-history?mango-account=${mangoAccountPk}&offset=${offset}&limit=${limit}`,
)
const parsedHistory = await history.json()
const sortedHistory =
parsedHistory && parsedHistory.length
? parsedHistory.sort(
(a: SwapHistoryItem, b: SwapHistoryItem) =>
dayjs(b.block_datetime).unix() -
dayjs(a.block_datetime).unix(),
)
: []
const combinedHistory =
offset !== 0
? loadedSwapHistory.concat(sortedHistory)
: sortedHistory
set((state) => {
state.mangoAccount.swapHistory.data = combinedHistory
})
} catch (e) {
console.error('Unable to fetch swap history', e)
} finally {
const notLoaded =
mangoStore.getState().mangoAccount.swapHistory.initialLoad
set((state) => {
state.mangoAccount.swapHistory.loading = false
if (notLoaded) {
state.mangoAccount.swapHistory.initialLoad = false
}
})
}
}, timeout)
},
fetchWalletTokens: async (walletPk: PublicKey) => {
const set = get().set
const connection = get().connection
if (walletPk) {
try {
const token = await getTokenAccountsByOwnerWithWrappedSol(
connection,
walletPk,
)
set((state) => {
state.wallet.tokens = token
})
} catch (e) {
notify({
title: 'Failed to refresh wallet balances.',
type: 'info',
})
}
} else {
set((state) => {
state.wallet.tokens = []
})
}
},
connectMangoClientWithWallet: async (wallet: WalletAdapter) => {
const set = get().set
try {
const provider = new AnchorProvider(
connection,
wallet.adapter as unknown as Wallet,
options,
)
const backupConnections = createBackupConnections(connection)
const priorityFee = get().priorityFee ?? DEFAULT_PRIORITY_FEE
const client = initMangoClient(
provider,
{
prioritizationFee: priorityFee,
prependedGlobalAdditionalInstructions:
get().prependedGlobalAdditionalInstructions,
multipleConnections: backupConnections,
},
null,
)
set((s) => {
s.client = client
})
} catch (e) {
if (e instanceof Error && e.name.includes('WalletLoadError')) {
notify({
title: `${wallet.adapter.name} Error`,
type: 'error',
description: `Please install ${wallet.adapter.name} and then reload this page.`,
})
}
}
},
async setPrependedGlobalAdditionalInstructions(
instructions: TransactionInstruction[],
) {
const set = get().set
const client = mangoStore.getState().client
const provider = client.program.provider as AnchorProvider
provider.opts.skipPreflight = true
const newClient = initMangoClient(
provider,
{
prioritizationFee: get().priorityFee,
prependedGlobalAdditionalInstructions: instructions,
multipleConnections: client.multipleConnections,
},
null,
)
set((s) => {
s.client = newClient
s.prependedGlobalAdditionalInstructions = instructions
})
},
async fetchTourSettings(walletPk: string) {
const set = get().set
set((state) => {
state.settings.loading = true
})
try {
const response = await fetch(
`${MANGO_DATA_API_URL}/user-data/settings-unsigned?wallet-pk=${walletPk}`,
)
const data = await response.json()
set((state) => {
state.settings.tours = data
state.settings.loading = false
})
} catch (e) {
console.error(e)
set((state) => {
state.settings.loading = false
})
}
},
async loadMarketFills() {
const set = get().set
const selectedMarket = get().selectedMarket.current
const group = get().group
const client = get().client
const connection = get().connection
try {
let serumMarket
let perpMarket
if (!group || !selectedMarket) return
if (selectedMarket instanceof Serum3Market) {
serumMarket = group.getSerum3ExternalMarket(
selectedMarket.serumMarketExternal,
)
} else {
perpMarket = selectedMarket
}
let loadedFills: (ParsedFillEvent | SerumEvent)[] = []
if (serumMarket) {
const serumFills = (await serumMarket.loadFills(
connection,
10000,
)) as SerumEvent[]
loadedFills = serumFills.filter((f) => !f?.eventFlags?.maker)
} else if (perpMarket) {
const perpFills = (await perpMarket.loadFills(
client,
)) as unknown as ParsedFillEvent[]
loadedFills = perpFills.reverse()
}
set((state) => {
state.selectedMarket.fills = loadedFills
})
} catch (err) {
console.error('Error fetching fills:', err)
}
},
updateConnection(endpointUrl) {
const set = get().set
const client = mangoStore.getState().client
const newConnection = new web3.Connection(
endpointUrl,
CONNECTION_COMMITMENT,
)
const oldProvider = client.program.provider as AnchorProvider
const newProvider = new AnchorProvider(
newConnection,
oldProvider.wallet,
options,
)
newProvider.opts.skipPreflight = true
const newClient = initMangoClient(
newProvider,
{
prependedGlobalAdditionalInstructions:
get().prependedGlobalAdditionalInstructions,
prioritizationFee: DEFAULT_PRIORITY_FEE,
multipleConnections: client.multipleConnections,
},
null,
)
set((state) => {
state.connection = newConnection
state.client = newClient
})
},
updateFee: async (feeMultiplier, fee, telemetry) => {
const set = get().set
const client = mangoStore.getState().client
const currentFee = get().priorityFee
const currentTelemetry = get().telemetry
const provider = client.program.provider as AnchorProvider
provider.opts.skipPreflight = true
//limit fee estimate to prevent error
const feeEstimate = Math.min(
Math.ceil(fee * feeMultiplier),
LAMPORTS_PER_SOL * 0.01,
)
if (currentFee !== feeEstimate || !currentTelemetry) {
const newClient = initMangoClient(
provider,
{
prioritizationFee: feeEstimate,
prependedGlobalAdditionalInstructions:
get().prependedGlobalAdditionalInstructions,
multipleConnections: client.multipleConnections,
},
telemetry,
)
set((state) => {
state.priorityFee = feeEstimate
state.client = newClient
state.telemetry = telemetry
})
}
},
},
}
}),
)
mangoStore.subscribe((state) => state.mangoAccount.current, spotBalancesUpdater)
mangoStore.subscribe(
(state) => state.mangoAccount.openOrderAccounts,
spotBalancesUpdater,
)
mangoStore.subscribe(
(state) => state.mangoAccount.current,
perpPositionsUpdater,
)
export default mangoStore