404 lines
12 KiB
TypeScript
404 lines
12 KiB
TypeScript
import create, { State } from 'zustand'
|
|
import produce from 'immer'
|
|
import { Market } from '@project-serum/serum'
|
|
import {
|
|
IDS,
|
|
Config,
|
|
MarketKind,
|
|
MerpsClient as MangoClient,
|
|
MerpsGroup as MangoGroup,
|
|
MerpsAccount as MarginAccount,
|
|
MarketConfig,
|
|
getMarketByBaseSymbolAndKind,
|
|
GroupConfig,
|
|
TokenConfig,
|
|
getTokenAccountsByOwnerWithWrappedSol,
|
|
getTokenByMint,
|
|
TokenAccount,
|
|
nativeToUi,
|
|
MerpsCache,
|
|
PerpMarket,
|
|
getAllMarkets,
|
|
getMultipleAccounts,
|
|
PerpMarketConfig,
|
|
SpotMarketConfig,
|
|
PerpMarketLayout,
|
|
} from '@blockworks-foundation/mango-client'
|
|
// import { SRM_DECIMALS } from '@project-serum/serum/lib/token-instructions'
|
|
import {
|
|
AccountInfo,
|
|
Commitment,
|
|
Connection,
|
|
PublicKey,
|
|
TokenAmount,
|
|
} from '@solana/web3.js'
|
|
import { EndpointInfo, WalletAdapter } from '../@types/types'
|
|
import { getWalletTokenInfo } from '../utils/tokens'
|
|
import {
|
|
chunks,
|
|
decodeAndLoadMarkets,
|
|
getOrderBookAccountInfos,
|
|
isDefined,
|
|
zipDict,
|
|
} from '../utils'
|
|
import { notify } from '../utils/notifications'
|
|
import useAllMarkets from '../hooks/useAllMarkets'
|
|
|
|
export const ENDPOINTS: EndpointInfo[] = [
|
|
{
|
|
name: 'mainnet-beta',
|
|
url: 'https://mango.rpcpool.com/',
|
|
websocket: 'https://mango.rpcpool.com/',
|
|
custom: false,
|
|
},
|
|
{
|
|
name: 'devnet',
|
|
url: 'https://api.devnet.solana.com',
|
|
websocket: 'https://api.devnet.solana.com',
|
|
custom: false,
|
|
},
|
|
]
|
|
|
|
type ClusterType = 'mainnet-beta' | 'devnet'
|
|
|
|
const CLUSTER = (process.env.NEXT_PUBLIC_CLUSTER as ClusterType) || 'devnet'
|
|
const ENDPOINT = ENDPOINTS.find((e) => e.name === CLUSTER)
|
|
export const DEFAULT_CONNECTION = new Connection(
|
|
ENDPOINT.url,
|
|
'processed' as Commitment
|
|
)
|
|
export const WEBSOCKET_CONNECTION = new Connection(
|
|
ENDPOINT.websocket,
|
|
'processed' as Commitment
|
|
)
|
|
|
|
const DEFAULT_MANGO_GROUP_NAME = 'merps_test_v2.1'
|
|
const DEFAULT_MANGO_GROUP_CONFIG = Config.ids().getGroup(
|
|
CLUSTER,
|
|
DEFAULT_MANGO_GROUP_NAME
|
|
)
|
|
const defaultMangoGroupIds = IDS['groups'].find(
|
|
(group) => group.name === DEFAULT_MANGO_GROUP_NAME
|
|
)
|
|
|
|
export const programId = new PublicKey(defaultMangoGroupIds.merpsProgramId)
|
|
export const serumProgramId = new PublicKey(defaultMangoGroupIds.serumProgramId)
|
|
const merpsGroupPk = new PublicKey(defaultMangoGroupIds.publicKey)
|
|
|
|
export const mangoClient = new MangoClient(DEFAULT_CONNECTION, programId)
|
|
|
|
export const INITIAL_STATE = {
|
|
WALLET: {
|
|
providerUrl: null,
|
|
connected: false,
|
|
current: null,
|
|
tokens: [],
|
|
srmAccountsForOwner: [],
|
|
contributedSrm: 0,
|
|
},
|
|
}
|
|
|
|
// an object with keys of Solana account addresses that we are
|
|
// subscribing to with connection.onAccountChange() in the
|
|
// useHydrateStore hook
|
|
interface AccountInfoList {
|
|
[key: string]: AccountInfo<Buffer>
|
|
}
|
|
|
|
export interface WalletToken {
|
|
account: TokenAccount
|
|
config: TokenConfig
|
|
uiBalance: number
|
|
}
|
|
|
|
export interface Orderbook {
|
|
bids: number[][]
|
|
asks: number[][]
|
|
}
|
|
|
|
interface MangoStore extends State {
|
|
notifications: Array<{
|
|
type: string
|
|
message: string
|
|
description?: string
|
|
txid?: string
|
|
}>
|
|
accountInfos: AccountInfoList
|
|
connection: {
|
|
cluster: ClusterType
|
|
current: Connection
|
|
websocket: Connection
|
|
endpoint: string
|
|
}
|
|
selectedMarket: {
|
|
config: MarketConfig
|
|
current: Market | PerpMarket | null
|
|
markPrice: number
|
|
kind: string
|
|
askInfo: AccountInfo<Buffer> | null
|
|
bidInfo: AccountInfo<Buffer> | null
|
|
orderBook: Orderbook
|
|
}
|
|
mangoGroups: Array<MangoGroup>
|
|
selectedMangoGroup: {
|
|
config: GroupConfig
|
|
name: string
|
|
current: MangoGroup | null
|
|
markets: {
|
|
[address: string]: Market | PerpMarket
|
|
}
|
|
rootBanks: any[]
|
|
cache: MerpsCache | null
|
|
}
|
|
marginAccounts: MarginAccount[]
|
|
selectedMarginAccount: {
|
|
current: MarginAccount | null
|
|
initialLoad: boolean
|
|
}
|
|
tradeForm: {
|
|
side: 'buy' | 'sell'
|
|
price: number | ''
|
|
baseSize: number | ''
|
|
quoteSize: number | ''
|
|
tradeType: 'Market' | 'Limit'
|
|
}
|
|
wallet: {
|
|
providerUrl: string
|
|
connected: boolean
|
|
current: WalletAdapter | undefined
|
|
tokens: WalletToken[]
|
|
srmAccountsForOwner: any[]
|
|
contributedSrm: number
|
|
}
|
|
settings: {
|
|
uiLocked: boolean
|
|
}
|
|
tradeHistory: any[]
|
|
set: (x: any) => void
|
|
actions: {
|
|
[key: string]: () => void
|
|
}
|
|
}
|
|
|
|
const useMangoStore = create<MangoStore>((set, get) => ({
|
|
notifications: [],
|
|
accountInfos: {},
|
|
connection: {
|
|
cluster: CLUSTER,
|
|
current: DEFAULT_CONNECTION,
|
|
websocket: WEBSOCKET_CONNECTION,
|
|
endpoint: ENDPOINT.url,
|
|
},
|
|
selectedMangoGroup: {
|
|
config: DEFAULT_MANGO_GROUP_CONFIG,
|
|
name: DEFAULT_MANGO_GROUP_NAME,
|
|
current: null,
|
|
markets: {},
|
|
rootBanks: [],
|
|
cache: null,
|
|
},
|
|
selectedMarket: {
|
|
config: getMarketByBaseSymbolAndKind(
|
|
DEFAULT_MANGO_GROUP_CONFIG,
|
|
'BTC',
|
|
'spot'
|
|
) as MarketConfig,
|
|
kind: 'spot',
|
|
current: null,
|
|
markPrice: 0,
|
|
askInfo: null,
|
|
bidInfo: null,
|
|
orderBook: { bids: [], asks: [] },
|
|
},
|
|
mangoGroups: [],
|
|
marginAccounts: [],
|
|
selectedMarginAccount: {
|
|
current: null,
|
|
initialLoad: false,
|
|
},
|
|
tradeForm: {
|
|
side: 'buy',
|
|
baseSize: '',
|
|
quoteSize: '',
|
|
tradeType: 'Limit',
|
|
price: '',
|
|
},
|
|
wallet: INITIAL_STATE.WALLET,
|
|
settings: {
|
|
uiLocked: true,
|
|
},
|
|
tradeHistory: [],
|
|
set: (fn) => set(produce(fn)),
|
|
actions: {
|
|
async fetchWalletTokens() {
|
|
const groupConfig = get().selectedMangoGroup.config
|
|
const wallet = get().wallet.current
|
|
const connected = get().wallet.connected
|
|
const set = get().set
|
|
|
|
if (wallet?.publicKey && connected) {
|
|
const ownerAddress = wallet.publicKey
|
|
const ownedTokenAccounts = await getTokenAccountsByOwnerWithWrappedSol(
|
|
DEFAULT_CONNECTION,
|
|
ownerAddress
|
|
)
|
|
const tokens = []
|
|
ownedTokenAccounts.forEach((account) => {
|
|
const config = getTokenByMint(groupConfig, account.mint)
|
|
if (config) {
|
|
const uiBalance = nativeToUi(account.amount, config.decimals)
|
|
tokens.push({ account, config, uiBalance })
|
|
}
|
|
})
|
|
|
|
set((state) => {
|
|
state.wallet.tokens = tokens
|
|
})
|
|
} else {
|
|
set((state) => {
|
|
state.wallet.tokens = []
|
|
})
|
|
}
|
|
},
|
|
async fetchMarginAccounts() {
|
|
const mangoGroup = get().selectedMangoGroup.current
|
|
const selectedMarginAcount = get().selectedMarginAccount.current
|
|
const wallet = get().wallet.current
|
|
const set = get().set
|
|
|
|
if (!wallet?.publicKey || !wallet.publicKey) return
|
|
|
|
if (!selectedMarginAcount) {
|
|
set((state) => {
|
|
state.selectedMarginAccount.initialLoad = true
|
|
})
|
|
}
|
|
|
|
return mangoClient
|
|
.getMarginAccountsForOwner(mangoGroup, wallet.publicKey, true)
|
|
.then((marginAccounts) => {
|
|
if (marginAccounts.length > 0) {
|
|
const sortedAccounts = marginAccounts
|
|
.slice()
|
|
.sort((a, b) =>
|
|
a.publicKey.toBase58() > b.publicKey.toBase58() ? 1 : -1
|
|
)
|
|
console.log('margin acc: ', sortedAccounts[0])
|
|
|
|
set((state) => {
|
|
state.marginAccounts = sortedAccounts
|
|
state.selectedMarginAccount.current = sortedAccounts[0]
|
|
// if (state.selectedMarginAccount.current) {
|
|
// state.selectedMarginAccount.current = marginAccounts.find(
|
|
// (ma) =>
|
|
// ma.publicKey.equals(
|
|
// state.selectedMarginAccount.current.publicKey
|
|
// )
|
|
// )
|
|
// } else {
|
|
// const lastAccount = localStorage.getItem('lastAccountViewed')
|
|
|
|
// state.selectedMarginAccount.current =
|
|
// marginAccounts.find(
|
|
// (ma) => ma.publicKey.toString() === JSON.parse(lastAccount)
|
|
// ) || sortedAccounts[0]
|
|
// }
|
|
})
|
|
}
|
|
set((state) => {
|
|
state.selectedMarginAccount.initialLoad = false
|
|
})
|
|
})
|
|
.catch((err) => {
|
|
console.error('Could not get margin accounts for wallet', err)
|
|
})
|
|
},
|
|
async fetchMangoGroup() {
|
|
const set = get().set
|
|
const mangoGroupPk = merpsGroupPk
|
|
const mangoGroupConfig = get().selectedMangoGroup.config
|
|
const selectedMarketConfig = get().selectedMarket.config
|
|
|
|
return mangoClient
|
|
.getMerpsGroup(mangoGroupPk)
|
|
.then(async (mangoGroup) => {
|
|
// TODO also perps
|
|
const rootBanks = await mangoGroup.loadRootBanks(DEFAULT_CONNECTION)
|
|
const merpsCache = await mangoGroup.loadCache(DEFAULT_CONNECTION)
|
|
|
|
const allMarketConfigs = getAllMarkets(mangoGroupConfig);
|
|
const allMarketPks = allMarketConfigs.map(m => m.publicKey);
|
|
const allMarketAccountInfos = await getMultipleAccounts(DEFAULT_CONNECTION, allMarketPks);
|
|
const allMarketAccounts = allMarketConfigs.map((config, i) => {
|
|
if (config.kind == 'spot') {
|
|
const decoded = Market.getLayout(programId).decode(allMarketAccountInfos[i].accountInfo.data);
|
|
return new Market(decoded, config.baseDecimals, config.quoteDecimals, undefined, mangoGroupConfig.serumProgramId);
|
|
}
|
|
if (config.kind == 'perp') {
|
|
const decoded = PerpMarketLayout.decode(allMarketAccountInfos[i].accountInfo.data);
|
|
return new PerpMarket(config.publicKey, config.baseDecimals, config.quoteDecimals, decoded);
|
|
}
|
|
})
|
|
|
|
const allBidsAndAsksPks = allMarketConfigs.map(m => [m.bidsKey, m.asksKey]).flat()
|
|
const allBidsAndAsksAccountInfos = await getMultipleAccounts(DEFAULT_CONNECTION, allBidsAndAsksPks);
|
|
|
|
const allMarkets = zipDict(allMarketPks.map(pk => pk.toBase58()), allMarketAccounts);
|
|
console.log('all', allMarkets);
|
|
|
|
set((state) => {
|
|
state.selectedMangoGroup.current = mangoGroup
|
|
state.selectedMangoGroup.rootBanks = rootBanks
|
|
state.selectedMangoGroup.cache = merpsCache
|
|
state.selectedMangoGroup.markets = allMarkets
|
|
state.selectedMarket.current = allMarkets[selectedMarketConfig.publicKey.toBase58()]
|
|
|
|
allMarketAccountInfos
|
|
.concat(allBidsAndAsksAccountInfos)
|
|
.forEach(({ publicKey, accountInfo }) => {
|
|
console.log(publicKey.toBase58(), accountInfo)
|
|
state.accountInfos[publicKey.toBase58()] = accountInfo
|
|
})
|
|
})
|
|
})
|
|
.catch((err) => {
|
|
notify({
|
|
message: 'Could not get mango group: ',
|
|
description: `${err}`,
|
|
type: 'error',
|
|
})
|
|
console.log('Could not get mango group: ', err)
|
|
})
|
|
},
|
|
// async fetchTradeHistory(marginAccount = null) {
|
|
// const selectedMarginAccount =
|
|
// marginAccount || get().selectedMarginAccount.current
|
|
// const set = get().set
|
|
|
|
// if (!selectedMarginAccount) return
|
|
// if (selectedMarginAccount.openOrdersAccounts.length === 0) return
|
|
|
|
// const openOrdersAccounts =
|
|
// selectedMarginAccount.openOrdersAccounts.filter(isDefined)
|
|
// const publicKeys = openOrdersAccounts.map((act) =>
|
|
// act.publicKey.toString()
|
|
// )
|
|
// const results = await Promise.all(
|
|
// publicKeys.map(async (pk) => {
|
|
// const response = await fetch(
|
|
// `https://stark-fjord-45757.herokuapp.com/trades/open_orders/${pk.toString()}`
|
|
// )
|
|
|
|
// const parsedResponse = await response.json()
|
|
// return parsedResponse?.data ? parsedResponse.data : []
|
|
// })
|
|
// )
|
|
// set((state) => {
|
|
// state.tradeHistory = results
|
|
// })
|
|
// },
|
|
},
|
|
}))
|
|
|
|
export default useMangoStore
|