import { useEffect, useMemo, useState } from "react"; import { JsonRpcProvider, BlockTag } from "@ethersproject/providers"; import { Contract } from "@ethersproject/contracts"; import { BigNumber } from "@ethersproject/bignumber"; import AggregatorV3Interface from "@chainlink/contracts/abi/v0.8/AggregatorV3Interface.json"; import FeedRegistryInterface from "@chainlink/contracts/abi/v0.8/FeedRegistryInterface.json"; import { Fetcher } from "swr"; import useSWRImmutable from "swr/immutable"; import { ChecksummedAddress } from "./types"; const FEED_REGISTRY_MAINNET: ChecksummedAddress = "0x47Fb2585D2C56Fe188D0E6ec628a38b74fCeeeDf"; // The USD "token" address for Chainlink feed registry's purposes const USD = "0x0000000000000000000000000000000000000348"; type FeedRegistryFetcherKey = [ChecksummedAddress, BlockTag]; type FeedRegistryFetcherData = [BigNumber | undefined, number | undefined]; const feedRegistryFetcherKey = ( tokenAddress: ChecksummedAddress, blockTag: BlockTag | undefined ): FeedRegistryFetcherKey | null => { if (blockTag === undefined) { return null; } return [tokenAddress, blockTag]; }; const feedRegistryFetcher = ( provider: JsonRpcProvider | undefined ): Fetcher => async (tokenAddress, blockTag) => { // It work works on ethereum mainnet and kovan, see: // https://docs.chain.link/docs/feed-registry/ if (provider!.network.chainId !== 1) { throw new Error("FeedRegistry is supported only on mainnet"); } // Let SWR handle error const feedRegistry = new Contract( FEED_REGISTRY_MAINNET, FeedRegistryInterface, provider ); const priceData = await feedRegistry.latestRoundData(tokenAddress, USD, { blockTag, }); const quote = BigNumber.from(priceData.answer); const decimals = await feedRegistry.decimals(tokenAddress, USD, { blockTag, }); return [quote, decimals]; }; export const useTokenUSDOracle = ( provider: JsonRpcProvider | undefined, blockTag: BlockTag | undefined, tokenAddress: ChecksummedAddress ): [BigNumber | undefined, number | undefined] => { const fetcher = feedRegistryFetcher(provider); const { data, error } = useSWRImmutable( feedRegistryFetcherKey(tokenAddress, blockTag), fetcher ); if (error) { return [undefined, undefined]; } return data ?? [undefined, undefined]; }; export const useETHUSDOracle = ( provider: JsonRpcProvider | undefined, blockTag: BlockTag | undefined ) => { const blockTags = useMemo(() => [blockTag], [blockTag]); const priceMap = useMultipleETHUSDOracle(provider, blockTags); if (blockTag === undefined) { return undefined; } return priceMap[blockTag]; }; export const useMultipleETHUSDOracle = ( provider: JsonRpcProvider | undefined, blockTags: (BlockTag | undefined)[] ) => { const ethFeed = useMemo(() => { // TODO: it currently is hardcoded to support only mainnet if (!provider || provider.network.chainId !== 1) { return undefined; } try { return new Contract("eth-usd.data.eth", AggregatorV3Interface, provider); } catch (err) { console.error(err); return undefined; } }, [provider]); const [latestPriceData, setLatestPriceData] = useState< Record >({}); useEffect(() => { if (!ethFeed) { return; } const priceReaders: Promise[] = []; for (const blockTag of blockTags) { priceReaders.push( (async () => { try { const priceData = await ethFeed.latestRoundData({ blockTag }); return BigNumber.from(priceData.answer); } catch (err) { // Silently ignore on purpose; it means the network or block number does // not contain the chainlink feed contract return undefined; } })() ); } const readData = async () => { const results = await Promise.all(priceReaders); const priceMap: Record = {}; for (let i = 0; i < blockTags.length; i++) { const blockTag = blockTags[i]; const result = results[i]; if (blockTag === undefined || result === undefined) { continue; } priceMap[blockTag] = result; } setLatestPriceData(priceMap); }; readData(); }, [ethFeed, blockTags]); return latestPriceData; };