2021-04-02 11:26:21 -07:00
|
|
|
import create, { State } from 'zustand'
|
|
|
|
import produce from 'immer'
|
|
|
|
import { Market } from '@project-serum/serum'
|
|
|
|
import {
|
2021-04-06 15:11:42 -07:00
|
|
|
IDS,
|
2021-06-16 18:50:16 -07:00
|
|
|
MerpsClient as MangoClient,
|
|
|
|
MerpsGroup as MangoGroup,
|
|
|
|
MerpsAccount as MarginAccount,
|
2021-04-02 11:26:21 -07:00
|
|
|
} from '@blockworks-foundation/mango-client'
|
2021-06-16 18:50:16 -07:00
|
|
|
// import { SRM_DECIMALS } from '@project-serum/serum/lib/token-instructions'
|
2021-04-09 17:01:00 -07:00
|
|
|
import { AccountInfo, Connection, PublicKey } from '@solana/web3.js'
|
2021-04-13 22:23:50 -07:00
|
|
|
import { EndpointInfo, WalletAdapter } from '../@types/types'
|
2021-05-20 11:51:00 -07:00
|
|
|
import { getWalletTokenInfo } from '../utils/tokens'
|
2021-06-16 18:50:16 -07:00
|
|
|
// import { isDefined } from '../utils/index'
|
2021-04-20 07:09:25 -07:00
|
|
|
import { notify } from '../utils/notifications'
|
2021-04-05 07:32:11 -07:00
|
|
|
|
|
|
|
export const ENDPOINTS: EndpointInfo[] = [
|
|
|
|
{
|
|
|
|
name: 'mainnet-beta',
|
2021-05-04 12:44:53 -07:00
|
|
|
url: 'https://mango.rpcpool.com/',
|
|
|
|
websocket: 'https://mango.rpcpool.com/',
|
2021-04-05 07:32:11 -07:00
|
|
|
custom: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'devnet',
|
2021-06-11 07:41:45 -07:00
|
|
|
url: 'https://api.devnet.solana.com',
|
2021-06-12 12:10:56 -07:00
|
|
|
websocket: 'https://api.devnet.solana.com',
|
2021-04-05 07:32:11 -07:00
|
|
|
custom: false,
|
|
|
|
},
|
|
|
|
]
|
|
|
|
|
2021-04-25 09:22:28 -07:00
|
|
|
type ClusterType = 'mainnet-beta' | 'devnet'
|
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
const CLUSTER = (process.env.NEXT_PUBLIC_CLUSTER as ClusterType) || 'devnet'
|
2021-04-20 14:00:45 -07:00
|
|
|
const ENDPOINT = ENDPOINTS.find((e) => e.name === CLUSTER)
|
|
|
|
const DEFAULT_CONNECTION = new Connection(ENDPOINT.url, 'recent')
|
|
|
|
const WEBSOCKET_CONNECTION = new Connection(ENDPOINT.websocket, 'recent')
|
2021-06-16 18:50:16 -07:00
|
|
|
const DEFAULT_MANGO_GROUP_NAME = 'merps_test_v1'
|
|
|
|
|
|
|
|
const defaultMangoGroupIds = IDS['groups'].find(
|
|
|
|
(group) => group.name === DEFAULT_MANGO_GROUP_NAME
|
|
|
|
)
|
|
|
|
|
|
|
|
export const programId = new PublicKey(defaultMangoGroupIds.merps_program_id)
|
|
|
|
export const serumProgramId = new PublicKey(
|
|
|
|
defaultMangoGroupIds.serum_program_id
|
|
|
|
)
|
|
|
|
const merpsGroupPk = new PublicKey(defaultMangoGroupIds.key)
|
2021-04-05 07:32:11 -07:00
|
|
|
|
2021-04-13 16:41:04 -07:00
|
|
|
export const INITIAL_STATE = {
|
|
|
|
WALLET: {
|
2021-04-13 22:23:50 -07:00
|
|
|
providerUrl: null,
|
2021-04-13 16:41:04 -07:00
|
|
|
connected: false,
|
|
|
|
current: null,
|
|
|
|
balances: [],
|
|
|
|
srmAccountsForOwner: [],
|
|
|
|
contributedSrm: 0,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-04-10 14:12:15 -07:00
|
|
|
// an object with keys of Solana account addresses that we are
|
|
|
|
// subscribing to with connection.onAccountChange() in the
|
|
|
|
// useHydrateStore hook
|
2021-04-05 07:32:11 -07:00
|
|
|
interface AccountInfoList {
|
|
|
|
[key: string]: AccountInfo<Buffer>
|
|
|
|
}
|
2021-04-02 11:26:21 -07:00
|
|
|
|
|
|
|
interface MangoStore extends State {
|
2021-04-11 21:17:23 -07:00
|
|
|
notifications: Array<{
|
|
|
|
type: string
|
|
|
|
message: string
|
|
|
|
description?: string
|
|
|
|
txid?: string
|
|
|
|
}>
|
2021-04-05 07:32:11 -07:00
|
|
|
accountInfos: AccountInfoList
|
|
|
|
connection: {
|
2021-04-25 09:22:28 -07:00
|
|
|
cluster: ClusterType
|
2021-04-05 07:32:11 -07:00
|
|
|
current: Connection
|
2021-04-20 14:00:45 -07:00
|
|
|
websocket: Connection
|
2021-04-05 07:32:11 -07:00
|
|
|
endpoint: string
|
|
|
|
}
|
2021-04-29 07:38:28 -07:00
|
|
|
selectedMarket: {
|
|
|
|
name: string
|
|
|
|
address: string
|
2021-04-02 11:26:21 -07:00
|
|
|
current: Market | null
|
2021-04-06 15:11:42 -07:00
|
|
|
mangoProgramId: number | null
|
2021-04-02 11:26:21 -07:00
|
|
|
markPrice: number
|
|
|
|
orderBook: any[]
|
|
|
|
}
|
|
|
|
mangoClient: MangoClient
|
2021-04-05 07:32:11 -07:00
|
|
|
mangoGroups: Array<MangoGroup>
|
|
|
|
selectedMangoGroup: {
|
|
|
|
name: string
|
|
|
|
current: MangoGroup | null
|
2021-04-07 08:44:22 -07:00
|
|
|
markets: {
|
2021-04-29 08:08:36 -07:00
|
|
|
[address: string]: Market
|
2021-04-07 08:44:22 -07:00
|
|
|
}
|
2021-06-16 18:50:16 -07:00
|
|
|
ids: any
|
|
|
|
tokens: any[]
|
2021-04-05 07:32:11 -07:00
|
|
|
}
|
2021-04-24 19:10:28 -07:00
|
|
|
marginAccounts: MarginAccount[]
|
2021-04-05 07:32:11 -07:00
|
|
|
selectedMarginAccount: {
|
|
|
|
current: MarginAccount | null
|
2021-05-04 16:19:48 -07:00
|
|
|
initialLoad: boolean
|
2021-04-05 07:32:11 -07:00
|
|
|
}
|
2021-04-02 11:26:21 -07:00
|
|
|
tradeForm: {
|
2021-04-12 13:01:55 -07:00
|
|
|
side: 'buy' | 'sell'
|
|
|
|
price: number | ''
|
|
|
|
baseSize: number | ''
|
|
|
|
quoteSize: number | ''
|
|
|
|
tradeType: 'Market' | 'Limit'
|
2021-04-02 11:26:21 -07:00
|
|
|
}
|
2021-04-05 07:32:11 -07:00
|
|
|
wallet: {
|
2021-04-13 22:23:50 -07:00
|
|
|
providerUrl: string
|
2021-04-05 07:32:11 -07:00
|
|
|
connected: boolean
|
2021-04-13 22:23:50 -07:00
|
|
|
current: WalletAdapter | undefined
|
2021-04-09 17:01:00 -07:00
|
|
|
balances: Array<{ account: any; publicKey: PublicKey }>
|
2021-04-13 16:41:04 -07:00
|
|
|
srmAccountsForOwner: any[]
|
|
|
|
contributedSrm: number
|
2021-04-05 07:32:11 -07:00
|
|
|
}
|
2021-04-13 09:51:42 -07:00
|
|
|
settings: {
|
|
|
|
uiLocked: boolean
|
|
|
|
}
|
2021-04-14 15:46:36 -07:00
|
|
|
tradeHistory: any[]
|
2021-04-02 11:26:21 -07:00
|
|
|
set: (x: any) => void
|
2021-04-25 09:22:28 -07:00
|
|
|
actions: {
|
|
|
|
[key: string]: () => void
|
|
|
|
}
|
2021-04-02 11:26:21 -07:00
|
|
|
}
|
|
|
|
|
2021-04-12 20:39:08 -07:00
|
|
|
const useMangoStore = create<MangoStore>((set, get) => ({
|
|
|
|
notifications: [],
|
|
|
|
accountInfos: {},
|
|
|
|
connection: {
|
|
|
|
cluster: CLUSTER,
|
|
|
|
current: DEFAULT_CONNECTION,
|
2021-04-20 14:00:45 -07:00
|
|
|
websocket: WEBSOCKET_CONNECTION,
|
|
|
|
endpoint: ENDPOINT.url,
|
2021-04-12 20:39:08 -07:00
|
|
|
},
|
|
|
|
selectedMangoGroup: {
|
|
|
|
name: DEFAULT_MANGO_GROUP_NAME,
|
|
|
|
current: null,
|
|
|
|
markets: {},
|
2021-06-16 18:50:16 -07:00
|
|
|
ids: defaultMangoGroupIds,
|
|
|
|
tokens: defaultMangoGroupIds.tokens,
|
2021-04-12 20:39:08 -07:00
|
|
|
},
|
|
|
|
selectedMarket: {
|
2021-06-16 18:50:16 -07:00
|
|
|
name: defaultMangoGroupIds.spot_markets[0].base_symbol,
|
|
|
|
address: defaultMangoGroupIds.spot_markets[0].key,
|
2021-04-12 20:39:08 -07:00
|
|
|
current: null,
|
|
|
|
mangoProgramId: null,
|
|
|
|
markPrice: 0,
|
|
|
|
orderBook: [],
|
|
|
|
},
|
2021-06-16 18:50:16 -07:00
|
|
|
mangoClient: new MangoClient(DEFAULT_CONNECTION, programId),
|
2021-04-12 20:39:08 -07:00
|
|
|
mangoGroups: [],
|
|
|
|
marginAccounts: [],
|
|
|
|
selectedMarginAccount: {
|
|
|
|
current: null,
|
2021-05-04 16:19:48 -07:00
|
|
|
initialLoad: false,
|
2021-04-12 20:39:08 -07:00
|
|
|
},
|
|
|
|
tradeForm: {
|
|
|
|
side: 'buy',
|
|
|
|
baseSize: '',
|
|
|
|
quoteSize: '',
|
|
|
|
tradeType: 'Limit',
|
|
|
|
price: '',
|
|
|
|
},
|
2021-04-13 16:41:04 -07:00
|
|
|
wallet: INITIAL_STATE.WALLET,
|
2021-04-13 09:51:42 -07:00
|
|
|
settings: {
|
|
|
|
uiLocked: true,
|
|
|
|
},
|
2021-04-14 15:46:36 -07:00
|
|
|
tradeHistory: [],
|
2021-04-12 20:39:08 -07:00
|
|
|
set: (fn) => set(produce(fn)),
|
|
|
|
actions: {
|
|
|
|
async fetchWalletBalances() {
|
|
|
|
const connection = get().connection.current
|
|
|
|
const wallet = get().wallet.current
|
|
|
|
const connected = get().wallet.connected
|
|
|
|
const set = get().set
|
2021-04-14 15:46:36 -07:00
|
|
|
|
|
|
|
if (wallet?.publicKey && connected) {
|
2021-04-12 20:39:08 -07:00
|
|
|
const ownerAddress = wallet.publicKey
|
2021-05-20 11:51:00 -07:00
|
|
|
const ownedTokenAccounts = await getWalletTokenInfo(
|
2021-04-12 20:39:08 -07:00
|
|
|
connection,
|
|
|
|
ownerAddress
|
|
|
|
)
|
|
|
|
set((state) => {
|
|
|
|
state.wallet.balances = ownedTokenAccounts
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
set((state) => {
|
|
|
|
state.wallet.balances = []
|
|
|
|
})
|
|
|
|
}
|
2021-04-05 07:32:11 -07:00
|
|
|
},
|
2021-06-16 18:50:16 -07:00
|
|
|
// async fetchMangoSrmAccounts() {
|
|
|
|
// const connection = get().connection.current
|
|
|
|
// const wallet = get().wallet.current
|
|
|
|
// const connected = get().wallet.connected
|
|
|
|
// const selectedMangoGroup = get().selectedMangoGroup.current
|
|
|
|
// const cluster = get().connection.cluster
|
|
|
|
// const mangoClient = get().mangoClient
|
|
|
|
// const set = get().set
|
2021-04-12 20:39:08 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// if (wallet?.publicKey && connected && selectedMangoGroup) {
|
|
|
|
// const usersMangoSrmAccounts =
|
|
|
|
// await mangoClient.getMangoSrmAccountsForOwner(
|
|
|
|
// connection,
|
|
|
|
// new PublicKey(IDS[cluster].mango_program_id),
|
|
|
|
// selectedMangoGroup,
|
|
|
|
// wallet
|
|
|
|
// )
|
|
|
|
// if (usersMangoSrmAccounts.length) {
|
|
|
|
// set((state) => {
|
|
|
|
// state.wallet.srmAccountsForOwner = usersMangoSrmAccounts
|
|
|
|
// const totalSrmDeposits = usersMangoSrmAccounts.reduce(
|
|
|
|
// (prev, cur) => prev + nativeToUi(cur.amount, SRM_DECIMALS),
|
|
|
|
// 0
|
|
|
|
// )
|
|
|
|
// state.wallet.contributedSrm = totalSrmDeposits
|
|
|
|
// })
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// },
|
|
|
|
// async fetchMarginAccounts() {
|
|
|
|
// const connection = get().connection.current
|
|
|
|
// const mangoGroup = get().selectedMangoGroup.current
|
|
|
|
// const selectedMarginAcount = get().selectedMarginAccount.current
|
|
|
|
// const wallet = get().wallet.current
|
|
|
|
// const cluster = get().connection.cluster
|
|
|
|
// const mangoClient = get().mangoClient
|
|
|
|
// const programId = IDS[cluster].mango_program_id
|
|
|
|
// const set = get().set
|
2021-04-12 20:39:08 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// if (!wallet?.publicKey || !wallet.publicKey) return
|
2021-04-12 20:39:08 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// if (!selectedMarginAcount) {
|
|
|
|
// set((state) => {
|
|
|
|
// state.selectedMarginAccount.initialLoad = true
|
|
|
|
// })
|
|
|
|
// }
|
2021-05-04 16:19:48 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// return mangoClient
|
|
|
|
// .getMarginAccountsForOwner(
|
|
|
|
// connection,
|
|
|
|
// new PublicKey(programId),
|
|
|
|
// mangoGroup,
|
|
|
|
// wallet
|
|
|
|
// )
|
|
|
|
// .then((marginAccounts) => {
|
|
|
|
// if (marginAccounts.length > 0) {
|
|
|
|
// const sortedAccounts = marginAccounts
|
|
|
|
// .slice()
|
|
|
|
// .sort(
|
|
|
|
// (a, b) =>
|
|
|
|
// (a.publicKey.toBase58() > b.publicKey.toBase58() && 1) || -1
|
|
|
|
// )
|
|
|
|
// set((state) => {
|
|
|
|
// state.marginAccounts = sortedAccounts
|
|
|
|
// if (state.selectedMarginAccount.current) {
|
|
|
|
// state.selectedMarginAccount.current = marginAccounts.find(
|
|
|
|
// (ma) =>
|
|
|
|
// ma.publicKey.equals(
|
|
|
|
// state.selectedMarginAccount.current.publicKey
|
|
|
|
// )
|
|
|
|
// )
|
|
|
|
// } else {
|
|
|
|
// const lastAccount = localStorage.getItem('lastAccountViewed')
|
2021-06-07 13:55:10 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// 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)
|
|
|
|
// })
|
|
|
|
// },
|
2021-04-12 20:39:08 -07:00
|
|
|
async fetchMangoGroup() {
|
|
|
|
const mangoClient = get().mangoClient
|
|
|
|
const set = get().set
|
2021-06-16 18:50:16 -07:00
|
|
|
|
2021-04-12 20:39:08 -07:00
|
|
|
if (!mangoClient) return
|
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
const mangoGroupPk = merpsGroupPk
|
2021-04-12 20:39:08 -07:00
|
|
|
|
2021-04-20 14:52:13 -07:00
|
|
|
return mangoClient
|
2021-06-16 18:50:16 -07:00
|
|
|
.getMerpsGroup(mangoGroupPk)
|
2021-04-12 20:39:08 -07:00
|
|
|
.then(async (mangoGroup) => {
|
2021-06-16 18:50:16 -07:00
|
|
|
console.log('we have a mango group', mangoGroup)
|
|
|
|
|
|
|
|
// const srmAccountInfoPromise = connection.getAccountInfo(
|
|
|
|
// mangoGroup.srmVault
|
|
|
|
// )
|
|
|
|
// const pricesPromise = mangoGroup.getPrices(connection)
|
|
|
|
// const [srmAccountInfo, prices] = await Promise.all([
|
|
|
|
// srmAccountInfoPromise,
|
|
|
|
// pricesPromise,
|
|
|
|
// ])
|
2021-04-12 20:39:08 -07:00
|
|
|
// Set the mango group
|
2021-04-09 17:01:00 -07:00
|
|
|
set((state) => {
|
2021-04-12 20:39:08 -07:00
|
|
|
state.selectedMangoGroup.current = mangoGroup
|
2021-06-16 18:50:16 -07:00
|
|
|
// state.selectedMangoGroup.srmAccount = srmAccountInfo
|
|
|
|
// state.selectedMangoGroup.mintDecimals = mangoGroup.mintDecimals // TODO store "tokens" from merps group ids
|
|
|
|
// state.selectedMangoGroup.prices = prices
|
2021-04-09 17:01:00 -07:00
|
|
|
})
|
2021-04-12 20:39:08 -07:00
|
|
|
})
|
|
|
|
.catch((err) => {
|
2021-04-20 07:09:25 -07:00
|
|
|
notify({
|
|
|
|
message: 'Could not get mango group: ',
|
|
|
|
description: `${err}`,
|
|
|
|
type: 'error',
|
|
|
|
})
|
|
|
|
console.log('Could not get mango group: ', err)
|
2021-04-12 20:39:08 -07:00
|
|
|
})
|
2021-04-09 17:01:00 -07:00
|
|
|
},
|
2021-06-16 18:50:16 -07:00
|
|
|
// async fetchTradeHistory(marginAccount = null) {
|
|
|
|
// const selectedMarginAccount =
|
|
|
|
// marginAccount || get().selectedMarginAccount.current
|
|
|
|
// const set = get().set
|
2021-04-14 15:46:36 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// if (!selectedMarginAccount) return
|
|
|
|
// if (selectedMarginAccount.openOrdersAccounts.length === 0) return
|
2021-04-14 15:46:36 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// 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()}`
|
|
|
|
// )
|
2021-04-14 15:46:36 -07:00
|
|
|
|
2021-06-16 18:50:16 -07:00
|
|
|
// const parsedResponse = await response.json()
|
|
|
|
// return parsedResponse?.data ? parsedResponse.data : []
|
|
|
|
// })
|
|
|
|
// )
|
|
|
|
// set((state) => {
|
|
|
|
// state.tradeHistory = results
|
|
|
|
// })
|
|
|
|
// },
|
2021-04-12 20:39:08 -07:00
|
|
|
},
|
|
|
|
}))
|
2021-04-02 11:26:21 -07:00
|
|
|
|
|
|
|
export default useMangoStore
|