omega/ui/src/utils/accounts.tsx

706 lines
18 KiB
TypeScript

import React, { useCallback, useContext, useEffect, useState } from "react";
import { useConnection } from "./connection";
import { useWallet } from "./wallet";
import { AccountInfo, Connection, PublicKey } from "@solana/web3.js";
import { programIds, SWAP_HOST_FEE_ADDRESS, WRAPPED_SOL_MINT } from "./ids";
import { AccountLayout, u64, MintInfo, MintLayout } from "@solana/spl-token";
import { usePools } from "./pools";
import { TokenAccount, PoolInfo } from "./../models";
import { notify } from "./notifications";
import { chunks } from "./utils";
import { EventEmitter } from "./eventEmitter";
const AccountsContext = React.createContext<any>(null);
const accountEmitter = new EventEmitter();
const pendingMintCalls = new Map<string, Promise<MintInfo>>();
const mintCache = new Map<string, MintInfo>();
const pendingAccountCalls = new Map<string, Promise<TokenAccount>>();
const accountsCache = new Map<string, TokenAccount>();
const pendingCalls = new Map<string, Promise<ParsedAccountBase>>();
const genericCache = new Map<string, ParsedAccountBase>();
const getAccountInfo = async (connection: Connection, pubKey: PublicKey) => {
const info = await connection.getAccountInfo(pubKey);
if (info === null) {
throw new Error("Failed to find mint account");
}
return tokenAccountFactory(pubKey, info);
};
const getMintInfo = async (connection: Connection, pubKey: PublicKey) => {
const info = await connection.getAccountInfo(pubKey);
if (info === null) {
throw new Error("Failed to find mint account");
}
const data = Buffer.from(info.data);
return deserializeMint(data);
};
export interface ParsedAccountBase {
pubkey: PublicKey;
account: AccountInfo<Buffer>;
info: any; // TODO: change to unkown
}
export interface ParsedAccount<T> extends ParsedAccountBase {
info: T;
}
export type AccountParser = (
pubkey: PublicKey,
data: AccountInfo<Buffer>
) => ParsedAccountBase;
export const MintParser = (pubKey: PublicKey, info: AccountInfo<Buffer>) => {
const buffer = Buffer.from(info.data);
const data = deserializeMint(buffer);
const details = {
pubkey: pubKey,
account: {
...info,
},
info: data,
} as ParsedAccountBase;
return details;
};
export const TokenAccountParser = tokenAccountFactory;
export const GenericAccountParser = (
pubKey: PublicKey,
info: AccountInfo<Buffer>
) => {
const buffer = Buffer.from(info.data);
const details = {
pubkey: pubKey,
account: {
...info,
},
info: buffer,
} as ParsedAccountBase;
return details;
};
export const keyToAccountParser = new Map<string, AccountParser>();
export const cache = {
query: async (
connection: Connection,
pubKey: string | PublicKey,
parser?: AccountParser
) => {
let id: PublicKey;
if (typeof pubKey === "string") {
id = new PublicKey(pubKey);
} else {
id = pubKey;
}
const address = id.toBase58();
let account = genericCache.get(address);
if (account) {
return account;
}
let query = pendingCalls.get(address);
if (query) {
return query;
}
query = connection.getAccountInfo(id).then((data) => {
if (!data) {
throw new Error("Account not found");
}
return cache.add(id, data, parser);
}) as Promise<TokenAccount>;
pendingCalls.set(address, query as any);
return query;
},
add: (id: PublicKey, obj: AccountInfo<Buffer>, parser?: AccountParser) => {
const address = id.toBase58();
const deserialize = parser ? parser : keyToAccountParser.get(address);
if (!deserialize) {
throw new Error(
"Deserializer needs to be registered or passed as a parameter"
);
}
cache.registerParser(id, deserialize);
pendingCalls.delete(address);
const account = deserialize(id, obj);
genericCache.set(address, account);
return account;
},
get: (pubKey: string | PublicKey) => {
let key: string;
if (typeof pubKey !== "string") {
key = pubKey.toBase58();
} else {
key = pubKey;
}
return genericCache.get(key);
},
registerParser: (pubkey: PublicKey, parser: AccountParser) => {
keyToAccountParser.set(pubkey.toBase58(), parser);
},
queryAccount: async (connection: Connection, pubKey: string | PublicKey) => {
let id: PublicKey;
if (typeof pubKey === "string") {
id = new PublicKey(pubKey);
} else {
id = pubKey;
}
const address = id.toBase58();
let account = accountsCache.get(address);
if (account) {
return account;
}
let query = pendingAccountCalls.get(address);
if (query) {
return query;
}
query = getAccountInfo(connection, id).then((data) => {
pendingAccountCalls.delete(address);
accountsCache.set(address, data);
return data;
}) as Promise<TokenAccount>;
pendingAccountCalls.set(address, query as any);
return query;
},
addAccount: (pubKey: PublicKey, obj: AccountInfo<Buffer>) => {
const account = tokenAccountFactory(pubKey, obj);
accountsCache.set(account.pubkey.toBase58(), account);
return account;
},
deleteAccount: (pubkey: PublicKey) => {
const id = pubkey?.toBase58();
accountsCache.delete(id);
accountEmitter.raiseAccountUpdated(id);
},
getAccount: (pubKey: string | PublicKey) => {
let key: string;
if (typeof pubKey !== "string") {
key = pubKey.toBase58();
} else {
key = pubKey;
}
return accountsCache.get(key);
},
queryMint: async (connection: Connection, pubKey: string | PublicKey) => {
let id: PublicKey;
if (typeof pubKey === "string") {
id = new PublicKey(pubKey);
} else {
id = pubKey;
}
const address = id.toBase58();
let mint = mintCache.get(address);
if (mint) {
return mint;
}
let query = pendingMintCalls.get(address);
if (query) {
return query;
}
query = getMintInfo(connection, id).then((data) => {
pendingAccountCalls.delete(address);
mintCache.set(address, data);
return data;
}) as Promise<MintInfo>;
pendingAccountCalls.set(address, query as any);
return query;
},
getMint: (pubKey: string | PublicKey) => {
let key: string;
if (typeof pubKey !== "string") {
key = pubKey.toBase58();
} else {
key = pubKey;
}
return mintCache.get(key);
},
addMint: (pubKey: PublicKey, obj: AccountInfo<Buffer>) => {
const mint = deserializeMint(obj.data);
mintCache.set(pubKey.toBase58(), mint);
return mint;
},
};
export const getCachedAccount = (
predicate: (account: TokenAccount) => boolean
) => {
for (const account of accountsCache.values()) {
if (predicate(account)) {
return account as TokenAccount;
}
}
};
function tokenAccountFactory(pubKey: PublicKey, info: AccountInfo<Buffer>) {
const buffer = Buffer.from(info.data);
const data = deserializeAccount(buffer);
const details = {
pubkey: pubKey,
account: {
...info,
},
info: data,
} as TokenAccount;
return details;
}
function wrapNativeAccount(
pubkey: PublicKey,
account?: AccountInfo<Buffer>
): TokenAccount | undefined {
if (!account) {
return undefined;
}
return {
pubkey: pubkey,
account,
info: {
mint: WRAPPED_SOL_MINT,
owner: pubkey,
amount: new u64(account.lamports),
delegate: null,
delegatedAmount: new u64(0),
isInitialized: true,
isFrozen: false,
isNative: true,
rentExemptReserve: null,
closeAuthority: null,
},
};
}
const UseNativeAccount = () => {
const connection = useConnection();
const { wallet } = useWallet();
const [nativeAccount, setNativeAccount] = useState<AccountInfo<Buffer>>();
useEffect(() => {
if (!connection || !wallet?.publicKey) {
return;
}
connection.getAccountInfo(wallet.publicKey).then((acc) => {
if (acc) {
setNativeAccount(acc);
}
});
connection.onAccountChange(wallet.publicKey, (acc) => {
if (acc) {
setNativeAccount(acc);
}
});
}, [setNativeAccount, wallet, wallet.publicKey, connection]);
return { nativeAccount };
};
const PRECACHED_OWNERS = new Set<string>();
const precacheUserTokenAccounts = async (
connection: Connection,
owner?: PublicKey
) => {
if (!owner) {
return;
}
// used for filtering account updates over websocket
PRECACHED_OWNERS.add(owner.toBase58());
// user accounts are update via ws subscription
const accounts = await connection.getTokenAccountsByOwner(owner, {
programId: programIds().token,
});
accounts.value
.map((info) => {
const data = deserializeAccount(info.account.data);
// need to query for mint to get decimals
// TODO: move to web3.js for decoding on the client side... maybe with callback
const details = {
pubkey: info.pubkey,
account: {
...info.account,
},
info: data,
} as TokenAccount;
return details;
})
.forEach((acc) => {
accountsCache.set(acc.pubkey.toBase58(), acc);
});
};
export function AccountsProvider({ children = null as any }) {
const connection = useConnection();
const { wallet, connected } = useWallet();
const [tokenAccounts, setTokenAccounts] = useState<TokenAccount[]>([]);
const [userAccounts, setUserAccounts] = useState<TokenAccount[]>([]);
const { nativeAccount } = UseNativeAccount();
const { pools } = usePools();
const publicKey = wallet?.publicKey;
const selectUserAccounts = useCallback(() => {
return [...accountsCache.values()].filter(
(a) => a.info.owner.toBase58() === publicKey?.toBase58()
);
}, [publicKey]);
useEffect(() => {
setUserAccounts(
[
wrapNativeAccount(publicKey, nativeAccount),
...tokenAccounts,
].filter((a) => a !== undefined) as TokenAccount[]
);
}, [nativeAccount, publicKey, tokenAccounts]);
useEffect(() => {
if (!connection || !publicKey) {
setTokenAccounts([]);
} else {
// cache host accounts to avoid query during swap
precacheUserTokenAccounts(connection, SWAP_HOST_FEE_ADDRESS);
precacheUserTokenAccounts(connection, publicKey).then(() => {
setTokenAccounts(selectUserAccounts());
});
const dispose = accountEmitter.onAccount(() => {
setTokenAccounts(selectUserAccounts());
})
// This can return different types of accounts: token-account, mint, multisig
// TODO: web3.js expose ability to filter. discuss filter syntax
const tokenSubID = connection.onProgramAccountChange(
programIds().token,
(info) => {
// TODO: fix type in web3.js
const id = (info.accountId as unknown) as string;
// TODO: do we need a better way to identify layout (maybe a enum identifing type?)
if (info.accountInfo.data.length === AccountLayout.span) {
const data = deserializeAccount(info.accountInfo.data);
// TODO: move to web3.js for decoding on the client side... maybe with callback
const details = {
pubkey: new PublicKey((info.accountId as unknown) as string),
account: {
...info.accountInfo,
},
info: data,
} as TokenAccount;
if (
PRECACHED_OWNERS.has(details.info.owner.toBase58()) ||
accountsCache.has(id)
) {
accountsCache.set(id, details);
accountEmitter.raiseAccountUpdated(id);
}
} else if (info.accountInfo.data.length === MintLayout.span) {
if (mintCache.has(id)) {
const data = Buffer.from(info.accountInfo.data);
const mint = deserializeMint(data);
mintCache.set(id, mint);
}
accountEmitter.raiseAccountUpdated(id);
}
if (genericCache.has(id)) {
cache.add(new PublicKey(id), info.accountInfo);
}
},
"singleGossip"
);
return () => {
connection.removeProgramAccountChangeListener(tokenSubID);
dispose();
};
}
}, [connection, connected, publicKey, selectUserAccounts]);
return (
<AccountsContext.Provider
value={{
userAccounts,
pools,
nativeAccount,
}}
>
{children}
</AccountsContext.Provider>
);
}
export function useNativeAccount() {
const context = useContext(AccountsContext);
return {
account: context.nativeAccount as AccountInfo<Buffer>,
};
}
export const getMultipleAccounts = async (
connection: any,
keys: string[],
commitment: string
) => {
const result = await Promise.all(
chunks(keys, 99).map((chunk) =>
getMultipleAccountsCore(connection, chunk, commitment)
)
);
const array = result
.map(
(a) =>
a.array.filter(acc => !!acc).map((acc) => {
const { data, ...rest } = acc;
const obj = {
...rest,
data: Buffer.from(data[0], "base64"),
} as AccountInfo<Buffer>;
return obj;
}) as AccountInfo<Buffer>[]
)
.flat();
return { keys, array };
};
const getMultipleAccountsCore = async (
connection: any,
keys: string[],
commitment: string
) => {
const args = connection._buildArgs([keys], commitment, "base64");
const unsafeRes = await connection._rpcRequest("getMultipleAccounts", args);
if (unsafeRes.error) {
throw new Error(
"failed to get info about account " + unsafeRes.error.message
);
}
if (unsafeRes.result.value) {
const array = unsafeRes.result.value as AccountInfo<string[]>[];
return { keys, array };
}
// TODO: fix
throw new Error();
};
export function useMint(key?: string | PublicKey) {
const connection = useConnection();
const [mint, setMint] = useState<MintInfo>();
const id = typeof key === "string" ? key : key?.toBase58();
useEffect(() => {
if (!id) {
return;
}
cache
.queryMint(connection, id)
.then(setMint)
.catch((err) =>
notify({
message: err.message,
type: "error",
})
);
const dispose = accountEmitter.onAccount((e) => {
const event = e;
if (event.id === id) {
cache.queryMint(connection, id).then(setMint);
}
});
return () => {
dispose();
};
}, [connection, id]);
return mint;
}
export function useUserAccounts() {
const context = useContext(AccountsContext);
return {
userAccounts: context.userAccounts as TokenAccount[],
};
}
export function useAccount(pubKey?: PublicKey) {
const connection = useConnection();
const [account, setAccount] = useState<TokenAccount>();
const key = pubKey?.toBase58();
useEffect(() => {
const query = async () => {
try {
if (!key) {
return;
}
const acc = await cache.queryAccount(connection, key).catch((err) =>
notify({
message: err.message,
type: "error",
})
);
if (acc) {
setAccount(acc);
}
} catch (err) {
console.error(err);
}
};
query();
const dispose = accountEmitter.onAccount((e) => {
const event = e;
if (event.id === key) {
query();
}
});
return () => {
dispose();
};
}, [connection, key]);
return account;
}
export function useCachedPool() {
const context = useContext(AccountsContext);
return {
pools: context.pools as PoolInfo[],
};
}
export const useSelectedAccount = (account: string) => {
const { userAccounts } = useUserAccounts();
const index = userAccounts.findIndex(
(acc) => acc.pubkey.toBase58() === account
);
if (index !== -1) {
return userAccounts[index];
}
return;
};
export const useAccountByMint = (mint: string) => {
const { userAccounts } = useUserAccounts();
const index = userAccounts.findIndex(
(acc) => acc.info.mint.toBase58() === mint
);
if (index !== -1) {
return userAccounts[index];
}
return;
};
// TODO: expose in spl package
const deserializeAccount = (data: Buffer) => {
const accountInfo = AccountLayout.decode(data);
accountInfo.mint = new PublicKey(accountInfo.mint);
accountInfo.owner = new PublicKey(accountInfo.owner);
accountInfo.amount = u64.fromBuffer(accountInfo.amount);
if (accountInfo.delegateOption === 0) {
accountInfo.delegate = null;
accountInfo.delegatedAmount = new u64(0);
} else {
accountInfo.delegate = new PublicKey(accountInfo.delegate);
accountInfo.delegatedAmount = u64.fromBuffer(accountInfo.delegatedAmount);
}
accountInfo.isInitialized = accountInfo.state !== 0;
accountInfo.isFrozen = accountInfo.state === 2;
if (accountInfo.isNativeOption === 1) {
accountInfo.rentExemptReserve = u64.fromBuffer(accountInfo.isNative);
accountInfo.isNative = true;
} else {
accountInfo.rentExemptReserve = null;
accountInfo.isNative = false;
}
if (accountInfo.closeAuthorityOption === 0) {
accountInfo.closeAuthority = null;
} else {
accountInfo.closeAuthority = new PublicKey(accountInfo.closeAuthority);
}
return accountInfo;
};
// TODO: expose in spl package
const deserializeMint = (data: Buffer) => {
if (data.length !== MintLayout.span) {
throw new Error("Not a valid Mint");
}
const mintInfo = MintLayout.decode(data);
if (mintInfo.mintAuthorityOption === 0) {
mintInfo.mintAuthority = null;
} else {
mintInfo.mintAuthority = new PublicKey(mintInfo.mintAuthority);
}
mintInfo.supply = u64.fromBuffer(mintInfo.supply);
mintInfo.isInitialized = mintInfo.isInitialized !== 0;
if (mintInfo.freezeAuthorityOption === 0) {
mintInfo.freezeAuthority = null;
} else {
mintInfo.freezeAuthority = new PublicKey(mintInfo.freezeAuthority);
}
return mintInfo as MintInfo;
};