mango-ui-v3/utils/index.ts

291 lines
6.7 KiB
TypeScript
Raw Normal View History

import {
getMultipleAccounts,
GroupConfig,
} from '@blockworks-foundation/mango-client'
2021-06-17 11:38:53 -07:00
import { I80F48 } from '@blockworks-foundation/mango-client/lib/src/fixednum'
import { Market, TOKEN_MINTS } from '@project-serum/serum'
import { AccountInfo, PublicKey } from '@solana/web3.js'
import BN from 'bn.js'
import { DEFAULT_CONNECTION, Orderbook } from '../stores/useMangoStore'
export async function sleep(ms) {
return new Promise((resolve) => setTimeout(resolve, ms))
}
export const percentFormat = new Intl.NumberFormat(undefined, {
style: 'percent',
minimumFractionDigits: 2,
maximumFractionDigits: 2,
})
export function floorToDecimal(
value: number,
decimals: number | undefined | null
) {
return decimals
? Math.floor(value * 10 ** decimals) / 10 ** decimals
: Math.floor(value)
}
export function ceilToDecimal(
value: number,
decimals: number | undefined | null
) {
return decimals
? Math.ceil(value * 10 ** decimals) / 10 ** decimals
: Math.ceil(value)
}
export function roundToDecimal(
value: number,
decimals: number | undefined | null
) {
return decimals ? Math.round(value * 10 ** decimals) / 10 ** decimals : value
}
export function getDecimalCount(value): number {
if (
!isNaN(value) &&
Math.floor(value) !== value &&
value.toString().includes('.')
)
return value.toString().split('.')[1].length || 0
if (
!isNaN(value) &&
Math.floor(value) !== value &&
value.toString().includes('e')
)
return parseInt(value.toString().split('e-')[1] || '0')
return 0
}
export function getTokenMultiplierFromDecimals(decimals: number): BN {
return new BN(10).pow(new BN(decimals))
}
2021-04-09 17:01:00 -07:00
export function abbreviateAddress(address: PublicKey, size = 5) {
const base58 = address.toBase58()
return base58.slice(0, size) + '…' + base58.slice(-size)
}
export function isEqual(obj1, obj2, keys) {
if (!keys && Object.keys(obj1).length !== Object.keys(obj2).length) {
return false
}
keys = keys || Object.keys(obj1)
for (const k of keys) {
if (obj1[k] !== obj2[k]) {
// shallow comparison
return false
}
}
return true
}
export function groupBy(list, keyGetter) {
const map = new Map()
list.forEach((item) => {
const key = keyGetter(item)
const collection = map.get(key)
if (!collection) {
map.set(key, [item])
} else {
collection.push(item)
}
})
return map
}
export function isDefined<T>(argument: T | undefined): argument is T {
return argument !== undefined
}
2021-04-02 11:26:21 -07:00
2021-08-18 13:15:17 -07:00
export function calculateTradePrice(
tradeType: string,
orderBook: Orderbook,
baseSize: number,
side: 'buy' | 'sell',
price: string | number
): number {
if (tradeType === 'Market') {
return calculateMarketPrice(orderBook, baseSize, side)
}
return Number(price)
}
2021-04-02 11:26:21 -07:00
export const calculateMarketPrice = (
2021-06-17 17:32:45 -07:00
orderBook: Orderbook,
2021-04-02 11:26:21 -07:00
size: number,
2021-06-17 17:32:45 -07:00
side: 'buy' | 'sell'
2021-08-18 13:15:17 -07:00
): number => {
2021-06-17 17:32:45 -07:00
const orders = side === 'buy' ? orderBook.asks : orderBook.bids
2021-04-02 11:26:21 -07:00
let acc = 0
let selectedOrder
2021-06-17 17:32:45 -07:00
for (const order of orders) {
2021-04-02 11:26:21 -07:00
acc += order[1]
if (acc >= size) {
selectedOrder = order
break
}
}
2021-08-18 13:15:17 -07:00
if (!selectedOrder) {
console.error('Orderbook empty no market price available')
return
}
2021-04-02 11:26:21 -07:00
if (side === 'buy') {
return selectedOrder[0] * 1.05
} else {
return selectedOrder[0] * 0.95
}
}
2021-04-09 07:27:49 -07:00
// Precision for our mango group token
export const tokenPrecision = {
BTC: 4,
ETH: 3,
2021-08-20 06:17:02 -07:00
MNGO: 2,
SOL: 2,
SRM: 2,
2021-04-09 07:27:49 -07:00
USDC: 2,
USDT: 2,
}
2021-04-09 17:01:00 -07:00
export const getSymbolForTokenMintAddress = (address: string): string => {
2021-04-13 16:41:04 -07:00
if (address && address.length) {
2021-04-26 09:59:01 -07:00
return TOKEN_MINTS.find((m) => m.address.toString() === address)?.name || ''
2021-04-13 16:41:04 -07:00
} else {
return ''
}
2021-04-09 17:01:00 -07:00
}
export const capitalize = (s) => {
if (typeof s !== 'string') return ''
return s.charAt(0).toUpperCase() + s.slice(1)
}
2021-06-18 20:07:57 -07:00
export function* chunks(arr, n) {
for (let i = 0; i < arr.length; i += n) {
yield arr.slice(i, i + n)
}
}
export function zipDict<K extends string | number | symbol, V>(
keys: K[],
values: V[]
) {
const result: Partial<Record<K, V>> = {}
2021-06-18 20:07:57 -07:00
keys.forEach((key, index) => {
result[key] = values[index]
})
return result
}
export const copyToClipboard = (copyThis) => {
const el = document.createElement('textarea')
el.value = copyThis.toString()
document.body.appendChild(el)
el.select()
document.execCommand('copy')
document.body.removeChild(el)
}
// Truncate decimals without rounding
export const trimDecimals = (n, digits) => {
2021-06-05 09:14:34 -07:00
const step = Math.pow(10, digits || 0)
const temp = Math.trunc(step * n)
return temp / step
}
2021-06-17 11:38:53 -07:00
export const i80f48ToPercent = (value: I80F48) =>
value.mul(I80F48.fromNumber(100))
export async function decodeAndLoadMarkets(
groupConfig: GroupConfig,
marketAccountInfos
): Promise<{ [marketPk: string]: Market }> {
const markets = {}
for (let i = 0; i < marketAccountInfos.length; i++) {
const { publicKey, accountInfo } = marketAccountInfos[i]
const decodedAcc = await Market.getLayout(
2021-06-18 15:51:34 -07:00
groupConfig.serumProgramId
).decode(accountInfo.data)
const baseToken = groupConfig.tokens.find((token) =>
2021-06-18 15:57:13 -07:00
token.mintKey.equals(decodedAcc.baseMint)
)
const quoteToken = groupConfig.tokens.find((token) =>
2021-06-18 15:57:13 -07:00
token.mintKey.equals(decodedAcc.quoteMint)
)
const baseMintDecimals = baseToken.decimals
const quoteMintDecimals = quoteToken.decimals
markets[publicKey] = new Market(
decodedAcc,
baseMintDecimals,
quoteMintDecimals,
{},
2021-06-18 15:51:34 -07:00
groupConfig.serumProgramId
)
}
return markets
}
export async function getOrderBookAccountInfos(
serumProgramId,
spotMarketAccountInfos: AccountInfo<Buffer>[]
): Promise<
{
publicKey: PublicKey
accountInfo: AccountInfo<Buffer>
}[]
> {
const decodedMarkets = spotMarketAccountInfos.map((accountInfo) =>
Market.getLayout(serumProgramId).decode(accountInfo.data)
)
const orderBookPks = []
decodedMarkets.forEach((mkt) => {
orderBookPks.push(mkt.bids)
orderBookPks.push(mkt.asks)
})
return await getMultipleAccounts(DEFAULT_CONNECTION, orderBookPks)
}
2021-07-25 06:54:25 -07:00
export const usdFormatter = (value, decimals = 2) => {
if (decimals === 0) {
value = Math.abs(value)
}
return new Intl.NumberFormat('en-US', {
2021-08-15 06:31:59 -07:00
style: 'currency',
currency: 'USD',
minimumFractionDigits: decimals,
maximumFractionDigits: decimals,
}).format(value)
}
2021-08-15 06:31:59 -07:00
export const formatUsdValue = (value) => {
2021-08-23 07:14:03 -07:00
const precision =
value >= 1 || value <= -1
? 2
: value === 0 ||
(value > 0 && value < 0.001) ||
(value < 0 && value > -0.001)
2021-08-23 07:14:03 -07:00
? 0
: 4
2021-08-15 06:31:59 -07:00
return usdFormatter(value, precision)
}
2021-08-24 03:09:42 -07:00
export const countLeadingZeros = (x) => {
if (x % 1 == 0) {
return 0
} else {
return -1 - Math.floor(Math.log10(x % 1))
}
}