@talismn/balances-react 0.3.3 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +24 -0
- package/dist/declarations/src/hooks/index.d.ts +20 -0
- package/dist/declarations/src/hooks/useAllAddresses.d.ts +4 -0
- package/dist/declarations/src/hooks/useBalanceModules.d.ts +8 -0
- package/dist/declarations/src/hooks/useBalances.d.ts +3 -6
- package/dist/declarations/src/hooks/useBalancesHydrate.d.ts +2 -0
- package/dist/declarations/src/hooks/useBalancesStatus.d.ts +19 -0
- package/dist/declarations/src/hooks/useChainConnectors.d.ts +12 -0
- package/dist/declarations/src/hooks/useChaindata.d.ts +6 -23
- package/dist/declarations/src/hooks/useChains.d.ts +3 -0
- package/dist/declarations/src/hooks/useDbCache.d.ts +24 -0
- package/dist/declarations/src/hooks/useDbCacheSubscription.d.ts +13 -0
- package/dist/declarations/src/hooks/useEvmNetworks.d.ts +3 -0
- package/dist/declarations/src/hooks/useTokenRates.d.ts +3 -3
- package/dist/declarations/src/hooks/useTokens.d.ts +3 -0
- package/dist/declarations/src/hooks/useWithTestnets.d.ts +8 -0
- package/dist/declarations/src/index.d.ts +1 -0
- package/dist/declarations/src/util/index.d.ts +2 -0
- package/dist/declarations/src/util/provideContext.d.ts +9 -0
- package/dist/declarations/src/util/useMulticastSubscription.d.ts +16 -0
- package/dist/declarations/src/util/useSharedSubscription.d.ts +9 -0
- package/dist/talismn-balances-react.cjs.dev.js +628 -248
- package/dist/talismn-balances-react.cjs.prod.js +628 -248
- package/dist/talismn-balances-react.esm.js +606 -249
- package/package.json +14 -11
| @@ -1,16 +1,197 @@ | |
| 1 | 
            -
            import {  | 
| 1 | 
            +
            import { useContext, createContext, useState, useEffect, useMemo, useRef, useCallback } from 'react';
         | 
| 2 | 
            +
            import { jsx } from 'react/jsx-runtime';
         | 
| 2 3 | 
             
            import { ChainConnector } from '@talismn/chain-connector';
         | 
| 3 4 | 
             
            import { ChainConnectorEvm } from '@talismn/chain-connector-evm';
         | 
| 5 | 
            +
            import { connectionMetaDb } from '@talismn/connection-meta';
         | 
| 6 | 
            +
            import { ChaindataProviderExtension } from '@talismn/chaindata-provider-extension';
         | 
| 7 | 
            +
            import { db as db$1, balances, Balances } from '@talismn/balances';
         | 
| 8 | 
            +
            import { db, fetchTokenRates } from '@talismn/token-rates';
         | 
| 4 9 | 
             
            import { useLiveQuery } from 'dexie-react-hooks';
         | 
| 5 | 
            -
            import { useState, useEffect, useRef } from 'react';
         | 
| 6 10 | 
             
            import { useDebounce } from 'react-use';
         | 
| 11 | 
            +
            import md5 from 'blueimp-md5';
         | 
| 7 12 | 
             
            import anylogger from 'anylogger';
         | 
| 8 | 
            -
            import {  | 
| 9 | 
            -
             | 
| 13 | 
            +
            import { Subject, Observable, defer, shareReplay } from 'rxjs';
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            const provideContext = useProviderContext => {
         | 
| 16 | 
            +
              // automatic typing based on our hook's return type
         | 
| 17 | 
            +
             | 
| 18 | 
            +
              const Context = /*#__PURE__*/createContext({
         | 
| 19 | 
            +
                __provideContextInternalDefaultValue: true
         | 
| 20 | 
            +
              });
         | 
| 21 | 
            +
              const Provider = ({
         | 
| 22 | 
            +
                children,
         | 
| 23 | 
            +
                ...props
         | 
| 24 | 
            +
              }) => {
         | 
| 25 | 
            +
                const ctx = useProviderContext(props);
         | 
| 26 | 
            +
                return /*#__PURE__*/jsx(Context.Provider, {
         | 
| 27 | 
            +
                  value: ctx,
         | 
| 28 | 
            +
                  children: children
         | 
| 29 | 
            +
                });
         | 
| 30 | 
            +
              };
         | 
| 31 | 
            +
              const useProvidedContext = () => {
         | 
| 32 | 
            +
                const context = useContext(Context);
         | 
| 33 | 
            +
                if (typeof context === "object" && context && "__provideContextInternalDefaultValue" in context) throw new Error("This hook requires a provider to be present above it in the tree");
         | 
| 34 | 
            +
                return context;
         | 
| 35 | 
            +
              };
         | 
| 36 | 
            +
              const result = [Provider, useProvidedContext];
         | 
| 37 | 
            +
              return result;
         | 
| 38 | 
            +
            };
         | 
| 39 | 
            +
             | 
| 40 | 
            +
            const useAllAddressesProvider = () => useState([]);
         | 
| 41 | 
            +
            const [AllAddressesProvider, useAllAddresses] = provideContext(useAllAddressesProvider);
         | 
| 42 | 
            +
             | 
| 43 | 
            +
            function useChaindataProvider(options = {}) {
         | 
| 44 | 
            +
              const [onfinalityApiKey, setOnfinalityApiKey] = useState(options.onfinalityApiKey);
         | 
| 45 | 
            +
             | 
| 46 | 
            +
              // make sure we recreate provider only when the onfinalityApiKey changes
         | 
| 47 | 
            +
              useEffect(() => {
         | 
| 48 | 
            +
                if (options.onfinalityApiKey !== onfinalityApiKey) setOnfinalityApiKey(options.onfinalityApiKey);
         | 
| 49 | 
            +
              }, [options.onfinalityApiKey, onfinalityApiKey]);
         | 
| 50 | 
            +
              return useMemo(() => new ChaindataProviderExtension({
         | 
| 51 | 
            +
                onfinalityApiKey
         | 
| 52 | 
            +
              }), [onfinalityApiKey]);
         | 
| 53 | 
            +
            }
         | 
| 54 | 
            +
            const [ChaindataProvider, useChaindata] = provideContext(useChaindataProvider);
         | 
| 55 | 
            +
             | 
| 56 | 
            +
            function useChainConnectorsProvider(options) {
         | 
| 57 | 
            +
              const [onfinalityApiKey, setOnfinalityApiKey] = useState(options.onfinalityApiKey);
         | 
| 58 | 
            +
             | 
| 59 | 
            +
              // make sure we recreate provider only when the onfinalityApiKey changes
         | 
| 60 | 
            +
              useEffect(() => {
         | 
| 61 | 
            +
                if (options.onfinalityApiKey !== onfinalityApiKey) setOnfinalityApiKey(options.onfinalityApiKey);
         | 
| 62 | 
            +
              }, [options.onfinalityApiKey, onfinalityApiKey]);
         | 
| 63 | 
            +
             | 
| 64 | 
            +
              // chaindata dependency
         | 
| 65 | 
            +
              const chaindata = useChaindata();
         | 
| 66 | 
            +
             | 
| 67 | 
            +
              // substrate connector
         | 
| 68 | 
            +
              const substrate = useMemo(() => new ChainConnector(chaindata, connectionMetaDb), [chaindata]);
         | 
| 69 | 
            +
             | 
| 70 | 
            +
              // evm connector
         | 
| 71 | 
            +
              const evm = useMemo(() => new ChainConnectorEvm(chaindata, {
         | 
| 72 | 
            +
                onfinalityApiKey
         | 
| 73 | 
            +
              }), [chaindata, onfinalityApiKey]);
         | 
| 74 | 
            +
              return useMemo(() => ({
         | 
| 75 | 
            +
                substrate,
         | 
| 76 | 
            +
                evm
         | 
| 77 | 
            +
              }), [substrate, evm]);
         | 
| 78 | 
            +
            }
         | 
| 79 | 
            +
            const [ChainConnectorsProvider, useChainConnectors] = provideContext(useChainConnectorsProvider);
         | 
| 80 | 
            +
             | 
| 81 | 
            +
            const useBalanceModulesProvider = ({
         | 
| 82 | 
            +
              balanceModules
         | 
| 83 | 
            +
            }) => {
         | 
| 84 | 
            +
              const chainConnectors = useChainConnectors();
         | 
| 85 | 
            +
              const chaindataProvider = useChaindata();
         | 
| 86 | 
            +
              const hydrated = useMemo(() => chainConnectors.substrate && chainConnectors.evm && chaindataProvider ? balanceModules.map(mod => mod({
         | 
| 87 | 
            +
                chainConnectors,
         | 
| 88 | 
            +
                chaindataProvider
         | 
| 89 | 
            +
              })) : [], [balanceModules, chainConnectors, chaindataProvider]);
         | 
| 90 | 
            +
              return hydrated;
         | 
| 91 | 
            +
            };
         | 
| 92 | 
            +
            const [BalanceModulesProvider, useBalanceModules] = provideContext(useBalanceModulesProvider);
         | 
| 93 | 
            +
             | 
| 94 | 
            +
            const filterNoTestnet = ({
         | 
| 95 | 
            +
              isTestnet
         | 
| 96 | 
            +
            }) => isTestnet === false;
         | 
| 97 | 
            +
            const DEFAULT_VALUE = {
         | 
| 98 | 
            +
              chainsWithTestnets: [],
         | 
| 99 | 
            +
              chainsWithoutTestnets: [],
         | 
| 100 | 
            +
              evmNetworksWithTestnets: [],
         | 
| 101 | 
            +
              evmNetworksWithoutTestnets: [],
         | 
| 102 | 
            +
              tokensWithTestnets: [],
         | 
| 103 | 
            +
              tokensWithoutTestnets: [],
         | 
| 104 | 
            +
              chainsWithTestnetsMap: {},
         | 
| 105 | 
            +
              chainsWithoutTestnetsMap: {},
         | 
| 106 | 
            +
              evmNetworksWithTestnetsMap: {},
         | 
| 107 | 
            +
              evmNetworksWithoutTestnetsMap: {},
         | 
| 108 | 
            +
              tokensWithTestnetsMap: {},
         | 
| 109 | 
            +
              tokensWithoutTestnetsMap: {},
         | 
| 110 | 
            +
              tokenRatesMap: {},
         | 
| 111 | 
            +
              balances: []
         | 
| 112 | 
            +
            };
         | 
| 113 | 
            +
            const consolidateDbCache = (chainsMap, evmNetworksMap, tokensMap, tokenRates, allBalances) => {
         | 
| 114 | 
            +
              if (!chainsMap || !evmNetworksMap || !tokensMap || !tokenRates || !allBalances) return DEFAULT_VALUE;
         | 
| 115 | 
            +
             | 
| 116 | 
            +
              // BEGIN: temp hack to indicate that
         | 
| 117 | 
            +
              //          - EVM GLMR is a mirror of substrate GLMR
         | 
| 118 | 
            +
              //          - EVM MOVR is a mirror of substrate MOVR
         | 
| 119 | 
            +
              //          - EVM DEV is a mirror of substrate DEV
         | 
| 120 | 
            +
              //          - EVM ACA is a mirror of substrate ACA
         | 
| 121 | 
            +
              const mirrorTokenIds = {
         | 
| 122 | 
            +
                "1284-evm-native-glmr": "moonbeam-substrate-native-glmr",
         | 
| 123 | 
            +
                "1285-evm-native-movr": "moonriver-substrate-native-movr",
         | 
| 124 | 
            +
                "1287-evm-native-dev": "moonbase-alpha-testnet-substrate-native-dev",
         | 
| 125 | 
            +
                "787-evm-native-aca": "acala-substrate-native-aca"
         | 
| 126 | 
            +
              };
         | 
| 127 | 
            +
              Object.entries(mirrorTokenIds).filter(([mirrorToken]) => tokensMap[mirrorToken]).forEach(([mirrorToken, mirrorOf]) => tokensMap[mirrorToken].mirrorOf = mirrorOf);
         | 
| 128 | 
            +
              // END: temp hack
         | 
| 129 | 
            +
             | 
| 130 | 
            +
              const chainsWithTestnets = Object.values(chainsMap);
         | 
| 131 | 
            +
              const chainsWithoutTestnets = chainsWithTestnets.filter(filterNoTestnet);
         | 
| 132 | 
            +
              const chainsWithoutTestnetsMap = Object.fromEntries(chainsWithoutTestnets.map(network => [network.id, network]));
         | 
| 133 | 
            +
              const evmNetworksWithTestnets = Object.values(evmNetworksMap);
         | 
| 134 | 
            +
              const evmNetworksWithoutTestnets = evmNetworksWithTestnets.filter(filterNoTestnet);
         | 
| 135 | 
            +
              const evmNetworksWithoutTestnetsMap = Object.fromEntries(evmNetworksWithoutTestnets.map(network => [network.id, network]));
         | 
| 136 | 
            +
             | 
| 137 | 
            +
              // ensure that we have corresponding network for each token
         | 
| 138 | 
            +
              const tokensWithTestnets = Object.values(tokensMap).filter(token => token.chain && chainsMap[token.chain.id] || token.evmNetwork && evmNetworksMap[token.evmNetwork.id]);
         | 
| 139 | 
            +
              const tokensWithoutTestnets = tokensWithTestnets.filter(filterNoTestnet).filter(token => token.chain && chainsWithoutTestnetsMap[token.chain.id] || token.evmNetwork && evmNetworksWithoutTestnetsMap[token.evmNetwork.id]);
         | 
| 140 | 
            +
              const tokensWithTestnetsMap = Object.fromEntries(tokensWithTestnets.map(token => [token.id, token]));
         | 
| 141 | 
            +
              const tokensWithoutTestnetsMap = Object.fromEntries(tokensWithoutTestnets.map(token => [token.id, token]));
         | 
| 142 | 
            +
              const tokenRatesMap = Object.fromEntries(tokenRates.map(({
         | 
| 143 | 
            +
                tokenId,
         | 
| 144 | 
            +
                rates
         | 
| 145 | 
            +
              }) => [tokenId, rates]));
         | 
| 146 | 
            +
             | 
| 147 | 
            +
              // return only balances for which we have a token
         | 
| 148 | 
            +
              const balances = allBalances.filter(b => tokensWithTestnetsMap[b.tokenId]);
         | 
| 149 | 
            +
              return {
         | 
| 150 | 
            +
                chainsWithTestnets,
         | 
| 151 | 
            +
                chainsWithoutTestnets,
         | 
| 152 | 
            +
                evmNetworksWithTestnets,
         | 
| 153 | 
            +
                evmNetworksWithoutTestnets,
         | 
| 154 | 
            +
                tokensWithTestnets,
         | 
| 155 | 
            +
                tokensWithoutTestnets,
         | 
| 156 | 
            +
                chainsWithTestnetsMap: chainsMap,
         | 
| 157 | 
            +
                chainsWithoutTestnetsMap,
         | 
| 158 | 
            +
                evmNetworksWithTestnetsMap: evmNetworksMap,
         | 
| 159 | 
            +
                evmNetworksWithoutTestnetsMap,
         | 
| 160 | 
            +
                tokensWithTestnetsMap,
         | 
| 161 | 
            +
                tokensWithoutTestnetsMap,
         | 
| 162 | 
            +
                tokenRatesMap,
         | 
| 163 | 
            +
                balances
         | 
| 164 | 
            +
              };
         | 
| 165 | 
            +
            };
         | 
| 166 | 
            +
            const useDbCacheProvider = () => {
         | 
| 167 | 
            +
              const chaindataProvider = useChaindata();
         | 
| 168 | 
            +
              const chainList = useLiveQuery(() => chaindataProvider?.chains(), [chaindataProvider]);
         | 
| 169 | 
            +
              const evmNetworkList = useLiveQuery(() => chaindataProvider?.evmNetworks(), [chaindataProvider]);
         | 
| 170 | 
            +
              const tokenList = useLiveQuery(() => chaindataProvider?.tokens(), [chaindataProvider]);
         | 
| 171 | 
            +
              const tokenRates = useLiveQuery(() => db.tokenRates.toArray(), []);
         | 
| 172 | 
            +
              const rawBalances = useLiveQuery(() => db$1.balances.toArray(), []);
         | 
| 173 | 
            +
              const [dbData, setDbData] = useState(DEFAULT_VALUE);
         | 
| 174 | 
            +
             | 
| 175 | 
            +
              // debounce every 500ms to prevent hammering UI with updates
         | 
| 176 | 
            +
              useDebounce(() => {
         | 
| 177 | 
            +
                setDbData(consolidateDbCache(chainList, evmNetworkList, tokenList, tokenRates, rawBalances));
         | 
| 178 | 
            +
              }, 500, [chainList, evmNetworkList, tokenList, rawBalances, tokenRates]);
         | 
| 179 | 
            +
              const refInitialized = useRef(false);
         | 
| 180 | 
            +
             | 
| 181 | 
            +
              // force an update as soon as all datasources are fetched, so UI can display data ASAP
         | 
| 182 | 
            +
              useEffect(() => {
         | 
| 183 | 
            +
                if (!refInitialized.current && chainList && evmNetworkList && tokenList && tokenRates && rawBalances) {
         | 
| 184 | 
            +
                  setDbData(consolidateDbCache(chainList, evmNetworkList, tokenList, tokenRates, rawBalances));
         | 
| 185 | 
            +
                  refInitialized.current = true;
         | 
| 186 | 
            +
                }
         | 
| 187 | 
            +
              }, [chainList, evmNetworkList, rawBalances, tokenList, tokenRates]);
         | 
| 188 | 
            +
              return dbData;
         | 
| 189 | 
            +
            };
         | 
| 190 | 
            +
            const [DbCacheProvider, useDbCache] = provideContext(useDbCacheProvider);
         | 
| 10 191 |  | 
| 11 192 | 
             
            var packageJson = {
         | 
| 12 193 | 
             
            	name: "@talismn/balances-react",
         | 
| 13 | 
            -
            	version: "0. | 
| 194 | 
            +
            	version: "0.4.0",
         | 
| 14 195 | 
             
            	author: "Talisman",
         | 
| 15 196 | 
             
            	homepage: "https://talisman.xyz",
         | 
| 16 197 | 
             
            	license: "UNLICENSED",
         | 
| @@ -28,7 +209,7 @@ var packageJson = { | |
| 28 209 | 
             
            		"/dist"
         | 
| 29 210 | 
             
            	],
         | 
| 30 211 | 
             
            	engines: {
         | 
| 31 | 
            -
            		node: ">= | 
| 212 | 
            +
            		node: ">=18"
         | 
| 32 213 | 
             
            	},
         | 
| 33 214 | 
             
            	scripts: {
         | 
| 34 215 | 
             
            		test: "jest",
         | 
| @@ -41,11 +222,14 @@ var packageJson = { | |
| 41 222 | 
             
            		"@talismn/chain-connector-evm": "workspace:^",
         | 
| 42 223 | 
             
            		"@talismn/chaindata-provider": "workspace:^",
         | 
| 43 224 | 
             
            		"@talismn/chaindata-provider-extension": "workspace:^",
         | 
| 225 | 
            +
            		"@talismn/connection-meta": "workspace:^",
         | 
| 44 226 | 
             
            		"@talismn/token-rates": "workspace:^",
         | 
| 45 227 | 
             
            		anylogger: "^1.0.11",
         | 
| 46 | 
            -
            		 | 
| 47 | 
            -
            		 | 
| 48 | 
            -
            		"react- | 
| 228 | 
            +
            		"blueimp-md5": "2.19.0",
         | 
| 229 | 
            +
            		dexie: "^3.2.3",
         | 
| 230 | 
            +
            		"dexie-react-hooks": "^1.1.3",
         | 
| 231 | 
            +
            		"react-use": "^17.4.0",
         | 
| 232 | 
            +
            		rxjs: "^7.8.0"
         | 
| 49 233 | 
             
            	},
         | 
| 50 234 | 
             
            	devDependencies: {
         | 
| 51 235 | 
             
            		"@talismn/eslint-config": "workspace:^",
         | 
| @@ -72,139 +256,387 @@ var packageJson = { | |
| 72 256 |  | 
| 73 257 | 
             
            var log = anylogger(packageJson.name);
         | 
| 74 258 |  | 
| 75 | 
            -
            //  | 
| 76 | 
            -
             | 
| 77 | 
            -
              const [chaindataProvider, setChaindataProvider] = useState(null);
         | 
| 259 | 
            +
            // global data store containing all subscriptions
         | 
| 260 | 
            +
            const subscriptions = {};
         | 
| 78 261 |  | 
| 79 | 
            -
             | 
| 80 | 
            -
             | 
| 262 | 
            +
            /**
         | 
| 263 | 
            +
             * This hook ensures a subscription is created only once, and unsubscribe automatically as soon as there is no consumer to the hook
         | 
| 264 | 
            +
             * @param key key that is unique to the subscription's parameters
         | 
| 265 | 
            +
             * @param subscribe // subscribe function that will be shared by all consumers of the key
         | 
| 266 | 
            +
             */
         | 
| 267 | 
            +
            const useSharedSubscription = (key, subscribe) => {
         | 
| 268 | 
            +
              // create the rxJS subject if it doesn't exist
         | 
| 269 | 
            +
              if (!subscriptions[key]) subscriptions[key] = {
         | 
| 270 | 
            +
                subject: new Subject()
         | 
| 271 | 
            +
              };
         | 
| 81 272 | 
             
              useEffect(() => {
         | 
| 82 | 
            -
                 | 
| 83 | 
            -
             | 
| 84 | 
            -
                 | 
| 85 | 
            -
                 | 
| 86 | 
            -
             | 
| 87 | 
            -
             | 
| 88 | 
            -
                   | 
| 89 | 
            -
             | 
| 90 | 
            -
                     | 
| 91 | 
            -
             | 
| 92 | 
            -
             | 
| 93 | 
            -
             | 
| 94 | 
            -
             | 
| 95 | 
            -
                     | 
| 96 | 
            -
                     | 
| 273 | 
            +
                // subscribe to subject.
         | 
| 274 | 
            +
                // it won't change but we need to count subscribers, to unsubscribe main subscription when no more observers
         | 
| 275 | 
            +
                const s = subscriptions[key].subject.subscribe();
         | 
| 276 | 
            +
                return () => {
         | 
| 277 | 
            +
                  // unsubscribe from our local observable updates to prevent memory leaks
         | 
| 278 | 
            +
                  s.unsubscribe();
         | 
| 279 | 
            +
                  const {
         | 
| 280 | 
            +
                    subject,
         | 
| 281 | 
            +
                    unsubscribe
         | 
| 282 | 
            +
                  } = subscriptions[key];
         | 
| 283 | 
            +
                  if (!subject.observed && unsubscribe) {
         | 
| 284 | 
            +
                    log.debug(`[useSharedSubscription] unsubscribing ${key}`);
         | 
| 285 | 
            +
             | 
| 286 | 
            +
                    // unsubscribe from backend updates to prevent unnecessary network connections
         | 
| 287 | 
            +
                    unsubscribe();
         | 
| 288 | 
            +
                    delete subscriptions[key].unsubscribe;
         | 
| 97 289 | 
             
                  }
         | 
| 98 290 | 
             
                };
         | 
| 99 | 
            -
             | 
| 100 | 
            -
             | 
| 291 | 
            +
              }, [key]);
         | 
| 292 | 
            +
             | 
| 293 | 
            +
              // Initialize subscription
         | 
| 294 | 
            +
              useEffect(() => {
         | 
| 295 | 
            +
                const {
         | 
| 296 | 
            +
                  unsubscribe
         | 
| 297 | 
            +
                } = subscriptions[key];
         | 
| 298 | 
            +
                // launch the subscription if it's a new key
         | 
| 299 | 
            +
                if (!unsubscribe) {
         | 
| 300 | 
            +
                  const cb = subscribe();
         | 
| 301 | 
            +
                  log.debug(`[useSharedSubscription] subscribing ${key}`);
         | 
| 302 | 
            +
                  if (cb) subscriptions[key].unsubscribe = cb;
         | 
| 303 | 
            +
                  // this error should only happen when developping a new hook, let it bubble up
         | 
| 304 | 
            +
                  else throw new Error(`${key} subscribe did not return an unsubscribe callback`);
         | 
| 305 | 
            +
                }
         | 
| 306 | 
            +
              }, [key, subscribe]);
         | 
| 307 | 
            +
            };
         | 
| 308 | 
            +
             | 
| 309 | 
            +
            /**
         | 
| 310 | 
            +
             * Creates a subscription function that can be used to subscribe to a multicast observable created from an upstream source.
         | 
| 311 | 
            +
             *
         | 
| 312 | 
            +
             * An example of when this is useful is when we want to subscribe to some data from multiple components, but we only want
         | 
| 313 | 
            +
             * to actively keep that data hydrated when at least one component is subscribed to it.
         | 
| 314 | 
            +
             *
         | 
| 315 | 
            +
             * When the first component subscribes, the `upstream` function will be called. It should then set up a subscription and return a teardown function.
         | 
| 316 | 
            +
             * When subsequent components subscribe, they will be added to the existing subscription.
         | 
| 317 | 
            +
             * When the last component unsubscribes, the teardown function returned from the `upstream` function will be called.
         | 
| 318 | 
            +
             *
         | 
| 319 | 
            +
             * @param upstream A function that takes a "next" callback function as an argument, and returns either an unsubscribe function or void.
         | 
| 320 | 
            +
             * @returns A subscription function that can be used to subscribe to the multicast observable.
         | 
| 321 | 
            +
             */
         | 
| 322 | 
            +
            const useMulticastSubscription = upstream => {
         | 
| 323 | 
            +
              const subscribe = useMemo(() => createMulticastSubscription(upstream), [upstream]);
         | 
| 324 | 
            +
              return subscribe;
         | 
| 325 | 
            +
            };
         | 
| 326 | 
            +
            const createMulticastSubscription = upstream => {
         | 
| 327 | 
            +
              // Create an upstream observable using the provided function.
         | 
| 328 | 
            +
              const upstreamObservable = new Observable(subscriber => {
         | 
| 329 | 
            +
                const unsubscribe = upstream(val => subscriber.next(val));
         | 
| 101 330 | 
             
                return () => {
         | 
| 102 | 
            -
                   | 
| 331 | 
            +
                  typeof unsubscribe === "function" && unsubscribe();
         | 
| 103 332 | 
             
                };
         | 
| 104 | 
            -
              } | 
| 105 | 
            -
             | 
| 106 | 
            -
               | 
| 333 | 
            +
              });
         | 
| 334 | 
            +
             | 
| 335 | 
            +
              // Create a multicast observable from the upstream observable, using the shareReplay operator.
         | 
| 336 | 
            +
              const multicastObservable = defer(() => upstreamObservable).pipe(shareReplay({
         | 
| 337 | 
            +
                bufferSize: 1,
         | 
| 338 | 
            +
                refCount: true
         | 
| 339 | 
            +
              }));
         | 
| 340 | 
            +
             | 
| 341 | 
            +
              // Create a subscription function that subscribes to the multicast observable and returns an unsubscribe function.
         | 
| 342 | 
            +
              const subscribe = callback => {
         | 
| 343 | 
            +
                const subscription = multicastObservable.subscribe(callback);
         | 
| 344 | 
            +
                const unsubscribe = () => subscription.unsubscribe();
         | 
| 345 | 
            +
                return unsubscribe;
         | 
| 346 | 
            +
              };
         | 
| 347 | 
            +
              return subscribe;
         | 
| 348 | 
            +
            };
         | 
| 349 | 
            +
             | 
| 350 | 
            +
            const useWithTestnetsProvider = ({
         | 
| 351 | 
            +
              withTestnets
         | 
| 352 | 
            +
            }) => {
         | 
| 353 | 
            +
              return {
         | 
| 354 | 
            +
                withTestnets
         | 
| 355 | 
            +
              };
         | 
| 356 | 
            +
            };
         | 
| 357 | 
            +
            const [WithTestnetsProvider, useWithTestnets] = provideContext(useWithTestnetsProvider);
         | 
| 358 | 
            +
             | 
| 359 | 
            +
            /**
         | 
| 360 | 
            +
             * This hook is responsible for fetching the data used for balances and inserting it into the db.
         | 
| 361 | 
            +
             */
         | 
| 362 | 
            +
            const useDbCacheSubscription = subscribeTo => {
         | 
| 363 | 
            +
              const provider = useChaindata();
         | 
| 364 | 
            +
             | 
| 365 | 
            +
              // can't handle balances & tokenRates here as they have other dependencies, it would trigger to many subscriptions
         | 
| 366 | 
            +
              const subscribe = useCallback(() => {
         | 
| 367 | 
            +
                switch (subscribeTo) {
         | 
| 368 | 
            +
                  case "chains":
         | 
| 369 | 
            +
                    return subscribeChainDataHydrate(provider, "chains");
         | 
| 370 | 
            +
                  case "evmNetworks":
         | 
| 371 | 
            +
                    return subscribeChainDataHydrate(provider, "evmNetworks");
         | 
| 372 | 
            +
                  case "tokens":
         | 
| 373 | 
            +
                    return subscribeChainDataHydrate(provider, "tokens");
         | 
| 374 | 
            +
                }
         | 
| 375 | 
            +
              }, [provider, subscribeTo]);
         | 
| 376 | 
            +
              useSharedSubscription(subscribeTo, subscribe);
         | 
| 377 | 
            +
            };
         | 
| 378 | 
            +
             | 
| 379 | 
            +
            /**
         | 
| 380 | 
            +
             * This hook is responsible for fetching the data used for token rates and inserting it into the db.
         | 
| 381 | 
            +
             */
         | 
| 382 | 
            +
            function useDbCacheTokenRatesSubscription() {
         | 
| 383 | 
            +
              const {
         | 
| 384 | 
            +
                withTestnets
         | 
| 385 | 
            +
              } = useWithTestnets();
         | 
| 386 | 
            +
              const tokens = useTokens$1(withTestnets);
         | 
| 387 | 
            +
              const subscriptionKey = useMemo(
         | 
| 388 | 
            +
              // not super sexy but we need key to change based on this stuff
         | 
| 389 | 
            +
              () => {
         | 
| 390 | 
            +
                const key = Object.values(tokens ?? {}).map(({
         | 
| 391 | 
            +
                  id
         | 
| 392 | 
            +
                }) => id).sort().join();
         | 
| 393 | 
            +
                return `tokenRates-${md5(key)}`;
         | 
| 394 | 
            +
              }, [tokens]);
         | 
| 395 | 
            +
              const subscription = useCallback(() => {
         | 
| 396 | 
            +
                if (!Object.values(tokens ?? {}).length) return () => {};
         | 
| 397 | 
            +
                return subscribeTokenRates(tokens);
         | 
| 398 | 
            +
              }, [tokens]);
         | 
| 399 | 
            +
              useSharedSubscription(subscriptionKey, subscription);
         | 
| 107 400 | 
             
            }
         | 
| 108 | 
            -
             | 
| 109 | 
            -
             | 
| 110 | 
            -
             | 
| 111 | 
            -
             | 
| 112 | 
            -
             | 
| 113 | 
            -
             | 
| 114 | 
            -
             | 
| 115 | 
            -
             | 
| 401 | 
            +
             | 
| 402 | 
            +
            /**
         | 
| 403 | 
            +
             * This hook is responsible for fetching the data used for balances and inserting it into the db.
         | 
| 404 | 
            +
             */
         | 
| 405 | 
            +
            function useDbCacheBalancesSubscription() {
         | 
| 406 | 
            +
              const {
         | 
| 407 | 
            +
                withTestnets
         | 
| 408 | 
            +
              } = useWithTestnets();
         | 
| 409 | 
            +
              const balanceModules = useBalanceModules();
         | 
| 410 | 
            +
              const chaindataProvider = useChaindata();
         | 
| 411 | 
            +
              const chainConnectors = useChainConnectors();
         | 
| 412 | 
            +
              const [allAddresses] = useAllAddresses();
         | 
| 413 | 
            +
              const tokens = useTokens$1(withTestnets);
         | 
| 414 | 
            +
              const subscriptionKey = useMemo(
         | 
| 415 | 
            +
              // not super sexy but we need key to change based on this stuff
         | 
| 416 | 
            +
              () => {
         | 
| 417 | 
            +
                const key = allAddresses.sort().join().concat(...Object.values(tokens ?? {}).map(({
         | 
| 418 | 
            +
                  id
         | 
| 419 | 
            +
                }) => id).sort()).concat(`evm:${!!chainConnectors.evm}`, `sub:${!!chainConnectors.substrate}`, ...balanceModules.map(m => m.type).sort(), `cd:${!!chaindataProvider}`);
         | 
| 420 | 
            +
                return `balances-${md5(key)}`;
         | 
| 421 | 
            +
              }, [allAddresses, balanceModules, chainConnectors, chaindataProvider, tokens]);
         | 
| 422 | 
            +
              const subscription = useCallback(() => {
         | 
| 423 | 
            +
                if (!Object.values(tokens ?? {}).length || !allAddresses.length) return () => {};
         | 
| 424 | 
            +
                return subscribeBalances(tokens ?? {}, allAddresses, balanceModules);
         | 
| 425 | 
            +
              }, [allAddresses, balanceModules, tokens]);
         | 
| 426 | 
            +
              useSharedSubscription(subscriptionKey, subscription);
         | 
| 427 | 
            +
            }
         | 
| 428 | 
            +
             | 
| 429 | 
            +
            // subscriptionless version of useTokens, prevents circular dependency
         | 
| 430 | 
            +
            const useTokens$1 = withTestnets => {
         | 
| 431 | 
            +
              const {
         | 
| 432 | 
            +
                tokensWithTestnetsMap,
         | 
| 433 | 
            +
                tokensWithoutTestnetsMap
         | 
| 434 | 
            +
              } = useDbCache();
         | 
| 435 | 
            +
              return withTestnets ? tokensWithTestnetsMap : tokensWithoutTestnetsMap;
         | 
| 436 | 
            +
            };
         | 
| 437 | 
            +
            const subscribeChainDataHydrate = (provider, type) => {
         | 
| 438 | 
            +
              const chaindata = provider;
         | 
| 439 | 
            +
              const delay = 300_000; // 300_000ms = 300s = 5 minutes
         | 
| 440 | 
            +
             | 
| 441 | 
            +
              let timeout = null;
         | 
| 442 | 
            +
              const hydrate = async () => {
         | 
| 443 | 
            +
                try {
         | 
| 444 | 
            +
                  if (type === "chains") await chaindata.hydrateChains();
         | 
| 445 | 
            +
                  if (type === "evmNetworks") await chaindata.hydrateEvmNetworks();
         | 
| 446 | 
            +
                  if (type === "tokens") await chaindata.hydrateTokens();
         | 
| 447 | 
            +
                  timeout = setTimeout(hydrate, delay);
         | 
| 448 | 
            +
                } catch (error) {
         | 
| 449 | 
            +
                  const retryTimeout = 5_000; // 5_000ms = 5 seconds
         | 
| 450 | 
            +
                  log.error(`Failed to fetch chaindata, retrying in ${Math.round(retryTimeout / 1000)} seconds`, error);
         | 
| 451 | 
            +
                  timeout = setTimeout(hydrate, retryTimeout);
         | 
| 452 | 
            +
                }
         | 
| 453 | 
            +
              };
         | 
| 454 | 
            +
             | 
| 455 | 
            +
              // launch the loop
         | 
| 456 | 
            +
              hydrate();
         | 
| 457 | 
            +
              return () => {
         | 
| 458 | 
            +
                if (timeout) clearTimeout(timeout);
         | 
| 459 | 
            +
              };
         | 
| 460 | 
            +
            };
         | 
| 461 | 
            +
            const subscribeTokenRates = tokens => {
         | 
| 462 | 
            +
              const REFRESH_INTERVAL = 300_000; // 6 minutes
         | 
| 463 | 
            +
              const RETRY_INTERVAL = 5_000; // 5 sec
         | 
| 464 | 
            +
             | 
| 465 | 
            +
              let timeout = null;
         | 
| 466 | 
            +
              const refreshTokenRates = async () => {
         | 
| 467 | 
            +
                try {
         | 
| 468 | 
            +
                  if (timeout) clearTimeout(timeout);
         | 
| 469 | 
            +
                  const tokenRates = await fetchTokenRates(tokens);
         | 
| 470 | 
            +
                  const putTokenRates = Object.entries(tokenRates).map(([tokenId, rates]) => ({
         | 
| 471 | 
            +
                    tokenId,
         | 
| 472 | 
            +
                    rates
         | 
| 473 | 
            +
                  }));
         | 
| 474 | 
            +
                  db.transaction("rw", db.tokenRates, async () => await db.tokenRates.bulkPut(putTokenRates));
         | 
| 475 | 
            +
                  timeout = setTimeout(() => {
         | 
| 476 | 
            +
                    refreshTokenRates();
         | 
| 477 | 
            +
                  }, REFRESH_INTERVAL);
         | 
| 478 | 
            +
                } catch (error) {
         | 
| 479 | 
            +
                  log.error(`Failed to fetch tokenRates, retrying in ${Math.round(RETRY_INTERVAL / 1000)} seconds`, error);
         | 
| 480 | 
            +
                  setTimeout(async () => {
         | 
| 481 | 
            +
                    refreshTokenRates();
         | 
| 482 | 
            +
                  }, RETRY_INTERVAL);
         | 
| 483 | 
            +
                }
         | 
| 484 | 
            +
              };
         | 
| 485 | 
            +
             | 
| 486 | 
            +
              // launch the loop
         | 
| 487 | 
            +
              refreshTokenRates();
         | 
| 488 | 
            +
              return () => {
         | 
| 489 | 
            +
                if (timeout) clearTimeout(timeout);
         | 
| 490 | 
            +
              };
         | 
| 491 | 
            +
            };
         | 
| 492 | 
            +
            const subscribeBalances = (tokens, addresses, balanceModules) => {
         | 
| 493 | 
            +
              const tokenIds = Object.values(tokens).map(({
         | 
| 494 | 
            +
                id
         | 
| 495 | 
            +
              }) => id);
         | 
| 496 | 
            +
              const addressesByToken = Object.fromEntries(tokenIds.map(tokenId => [tokenId, addresses]));
         | 
| 497 | 
            +
              const updateDb = balances => {
         | 
| 498 | 
            +
                const putBalances = Object.entries(balances.toJSON()).map(([id, balance]) => ({
         | 
| 499 | 
            +
                  id,
         | 
| 500 | 
            +
                  ...balance
         | 
| 501 | 
            +
                }));
         | 
| 502 | 
            +
                db$1.transaction("rw", db$1.balances, async () => await db$1.balances.bulkPut(putBalances));
         | 
| 503 | 
            +
              };
         | 
| 504 | 
            +
              let unsubscribed = false;
         | 
| 505 | 
            +
             | 
| 506 | 
            +
              // eslint-disable-next-line no-console
         | 
| 507 | 
            +
              log.log("subscribing to balance changes for %d tokens and %d addresses", tokenIds.length, addresses.length);
         | 
| 508 | 
            +
              const unsubs = balanceModules.map(async balanceModule => {
         | 
| 509 | 
            +
                // filter out tokens to only include those which this module knows how to fetch balances for
         | 
| 510 | 
            +
                const moduleTokenIds = Object.values(tokens ?? {}).filter(({
         | 
| 511 | 
            +
                  type
         | 
| 512 | 
            +
                }) => type === balanceModule.type).map(({
         | 
| 513 | 
            +
                  id
         | 
| 514 | 
            +
                }) => id);
         | 
| 515 | 
            +
                const addressesByModuleToken = Object.fromEntries(Object.entries(addressesByToken).filter(([tokenId]) => moduleTokenIds.includes(tokenId)));
         | 
| 516 | 
            +
                const unsub = balances(balanceModule, addressesByModuleToken, (error, balances) => {
         | 
| 517 | 
            +
                  // log errors
         | 
| 518 | 
            +
                  if (error) {
         | 
| 519 | 
            +
                    if (error?.type === "STALE_RPC_ERROR") return db$1.balances.where({
         | 
| 520 | 
            +
                      source: balanceModule.type,
         | 
| 521 | 
            +
                      chainId: error.chainId
         | 
| 522 | 
            +
                    }).filter(balance => {
         | 
| 523 | 
            +
                      if (!Object.keys(addressesByModuleToken).includes(balance.tokenId)) return false;
         | 
| 524 | 
            +
                      if (!addressesByModuleToken[balance.tokenId].includes(balance.address)) return false;
         | 
| 525 | 
            +
                      return true;
         | 
| 526 | 
            +
                    }).modify({
         | 
| 527 | 
            +
                      status: "stale"
         | 
| 528 | 
            +
                    });
         | 
| 529 | 
            +
                    return log.error(`Failed to fetch ${balanceModule.type} balances`, error);
         | 
| 530 | 
            +
                  }
         | 
| 531 | 
            +
                  // ignore empty balance responses
         | 
| 532 | 
            +
                  if (!balances) return;
         | 
| 533 | 
            +
                  // ignore balances from old subscriptions which are still in the process of unsubscribing
         | 
| 534 | 
            +
                  if (unsubscribed) return;
         | 
| 535 | 
            +
                  updateDb(balances);
         | 
| 116 536 | 
             
                });
         | 
| 117 | 
            -
             | 
| 118 | 
            -
             | 
| 537 | 
            +
                return () => {
         | 
| 538 | 
            +
                  // wait 2 seconds before actually unsubscribing, allowing for websocket to be reused
         | 
| 539 | 
            +
                  unsub.then(unsubscribe => {
         | 
| 540 | 
            +
                    setTimeout(unsubscribe, 2_000);
         | 
| 541 | 
            +
                  });
         | 
| 542 | 
            +
                  db$1.balances.where({
         | 
| 543 | 
            +
                    source: balanceModule.type
         | 
| 544 | 
            +
                  }).filter(balance => {
         | 
| 545 | 
            +
                    if (!Object.keys(addressesByModuleToken).includes(balance.tokenId)) return false;
         | 
| 546 | 
            +
                    if (!addressesByModuleToken[balance.tokenId].includes(balance.address)) return false;
         | 
| 547 | 
            +
                    return true;
         | 
| 548 | 
            +
                  }).modify({
         | 
| 549 | 
            +
                    status: "cache"
         | 
| 550 | 
            +
                  });
         | 
| 551 | 
            +
                };
         | 
| 552 | 
            +
              });
         | 
| 553 | 
            +
              const unsubscribeAll = () => {
         | 
| 554 | 
            +
                unsubscribed = true;
         | 
| 555 | 
            +
                unsubs.forEach(unsub => unsub.then(unsubscribe => unsubscribe()));
         | 
| 556 | 
            +
              };
         | 
| 557 | 
            +
              return unsubscribeAll;
         | 
| 558 | 
            +
            };
         | 
| 559 | 
            +
             | 
| 560 | 
            +
            function useChains(withTestnets) {
         | 
| 561 | 
            +
              // keep db data up to date
         | 
| 562 | 
            +
              useDbCacheSubscription("chains");
         | 
| 563 | 
            +
              const {
         | 
| 564 | 
            +
                chainsWithTestnetsMap,
         | 
| 565 | 
            +
                chainsWithoutTestnetsMap
         | 
| 566 | 
            +
              } = useDbCache();
         | 
| 567 | 
            +
              return withTestnets ? chainsWithTestnetsMap : chainsWithoutTestnetsMap;
         | 
| 119 568 | 
             
            }
         | 
| 120 | 
            -
            function useChain( | 
| 121 | 
            -
              const  | 
| 122 | 
            -
               | 
| 123 | 
            -
                if (chaindata === null) return;
         | 
| 124 | 
            -
                if (!chainId) return;
         | 
| 125 | 
            -
                chaindata.getChain(chainId).then(setChain);
         | 
| 126 | 
            -
              }, [chainId, chaindata, chaindata?.generation]);
         | 
| 127 | 
            -
              return chain;
         | 
| 569 | 
            +
            function useChain(chainId, withTestnets) {
         | 
| 570 | 
            +
              const chains = useChains(withTestnets);
         | 
| 571 | 
            +
              return chainId ? chains[chainId] : undefined;
         | 
| 128 572 | 
             
            }
         | 
| 129 | 
            -
             | 
| 130 | 
            -
             | 
| 131 | 
            -
               | 
| 132 | 
            -
             | 
| 133 | 
            -
             | 
| 134 | 
            -
                 | 
| 135 | 
            -
             | 
| 136 | 
            -
             | 
| 137 | 
            -
             | 
| 138 | 
            -
              }, [chaindata, chaindata?.generation]);
         | 
| 139 | 
            -
              return evmNetworks || {};
         | 
| 573 | 
            +
             | 
| 574 | 
            +
            function useEvmNetworks(withTestnets) {
         | 
| 575 | 
            +
              // keep db data up to date
         | 
| 576 | 
            +
              useDbCacheSubscription("evmNetworks");
         | 
| 577 | 
            +
              const {
         | 
| 578 | 
            +
                evmNetworksWithTestnetsMap,
         | 
| 579 | 
            +
                evmNetworksWithoutTestnetsMap
         | 
| 580 | 
            +
              } = useDbCache();
         | 
| 581 | 
            +
              return withTestnets ? evmNetworksWithTestnetsMap : evmNetworksWithoutTestnetsMap;
         | 
| 140 582 | 
             
            }
         | 
| 141 | 
            -
            function useEvmNetwork( | 
| 142 | 
            -
              const  | 
| 143 | 
            -
               | 
| 144 | 
            -
                if (chaindata === null) return;
         | 
| 145 | 
            -
                if (!evmNetworkId) return;
         | 
| 146 | 
            -
                chaindata.getEvmNetwork(evmNetworkId).then(setEvmNetwork);
         | 
| 147 | 
            -
              }, [chaindata, chaindata?.generation, evmNetworkId]);
         | 
| 148 | 
            -
              return evmNetwork;
         | 
| 583 | 
            +
            function useEvmNetwork(evmNetworkId, withTestnets) {
         | 
| 584 | 
            +
              const evmNetworks = useEvmNetworks(withTestnets);
         | 
| 585 | 
            +
              return evmNetworkId ? evmNetworks[evmNetworkId] : undefined;
         | 
| 149 586 | 
             
            }
         | 
| 150 | 
            -
             | 
| 151 | 
            -
             | 
| 152 | 
            -
               | 
| 153 | 
            -
             | 
| 154 | 
            -
             | 
| 155 | 
            -
                 | 
| 156 | 
            -
             | 
| 157 | 
            -
             | 
| 158 | 
            -
                });
         | 
| 159 | 
            -
              }, [chaindata, chaindata?.generation]);
         | 
| 160 | 
            -
              return tokens || {};
         | 
| 587 | 
            +
             | 
| 588 | 
            +
            function useTokenRates() {
         | 
| 589 | 
            +
              // keep db data up to date
         | 
| 590 | 
            +
              useDbCacheTokenRatesSubscription();
         | 
| 591 | 
            +
              const {
         | 
| 592 | 
            +
                tokenRatesMap
         | 
| 593 | 
            +
              } = useDbCache();
         | 
| 594 | 
            +
              return tokenRatesMap;
         | 
| 161 595 | 
             
            }
         | 
| 162 | 
            -
            function  | 
| 163 | 
            -
              const  | 
| 164 | 
            -
               | 
| 165 | 
            -
                if (chaindata === null) return;
         | 
| 166 | 
            -
                if (!tokenId) return;
         | 
| 167 | 
            -
                chaindata.getToken(tokenId).then(setToken);
         | 
| 168 | 
            -
              }, [chaindata, chaindata?.generation, tokenId]);
         | 
| 169 | 
            -
              return token;
         | 
| 596 | 
            +
            function useTokenRate(tokenId) {
         | 
| 597 | 
            +
              const tokenRates = useTokenRates();
         | 
| 598 | 
            +
              return tokenId ? tokenRates[tokenId] : undefined;
         | 
| 170 599 | 
             
            }
         | 
| 171 600 |  | 
| 172 | 
            -
            function  | 
| 173 | 
            -
               | 
| 174 | 
            -
               | 
| 175 | 
            -
               | 
| 176 | 
            -
                 | 
| 177 | 
            -
                 | 
| 178 | 
            -
             | 
| 179 | 
            -
             | 
| 180 | 
            -
             | 
| 181 | 
            -
             | 
| 182 | 
            -
             | 
| 183 | 
            -
             | 
| 184 | 
            -
                  if (thisGeneration !== generation.current) return;
         | 
| 185 | 
            -
                  setTokenRates(tokenRates);
         | 
| 186 | 
            -
                });
         | 
| 187 | 
            -
              }, [tokens]);
         | 
| 188 | 
            -
              return tokenRates || {};
         | 
| 601 | 
            +
            function useTokens(withTestnets) {
         | 
| 602 | 
            +
              // keep db data up to date
         | 
| 603 | 
            +
              useDbCacheSubscription("tokens");
         | 
| 604 | 
            +
              const {
         | 
| 605 | 
            +
                tokensWithTestnetsMap,
         | 
| 606 | 
            +
                tokensWithoutTestnetsMap
         | 
| 607 | 
            +
              } = useDbCache();
         | 
| 608 | 
            +
              return withTestnets ? tokensWithTestnetsMap : tokensWithoutTestnetsMap;
         | 
| 609 | 
            +
            }
         | 
| 610 | 
            +
            function useToken(tokenId, withTestnets) {
         | 
| 611 | 
            +
              const tokens = useTokens(withTestnets);
         | 
| 612 | 
            +
              return tokenId ? tokens[tokenId] : undefined;
         | 
| 189 613 | 
             
            }
         | 
| 190 614 |  | 
| 191 | 
            -
             | 
| 192 | 
            -
             | 
| 193 | 
            -
             | 
| 194 | 
            -
             | 
| 195 | 
            -
             | 
| 196 | 
            -
             | 
| 197 | 
            -
             | 
| 198 | 
            -
             | 
| 199 | 
            -
             | 
| 200 | 
            -
             | 
| 201 | 
            -
             | 
| 202 | 
            -
             | 
| 203 | 
            -
             | 
| 204 | 
            -
               | 
| 205 | 
            -
             | 
| 206 | 
            -
             | 
| 207 | 
            -
             | 
| 615 | 
            +
            const useBalancesHydrate = () => {
         | 
| 616 | 
            +
              const {
         | 
| 617 | 
            +
                withTestnets
         | 
| 618 | 
            +
              } = useWithTestnets();
         | 
| 619 | 
            +
              const chains = useChains(withTestnets);
         | 
| 620 | 
            +
              const evmNetworks = useEvmNetworks(withTestnets);
         | 
| 621 | 
            +
              const tokens = useTokens(withTestnets);
         | 
| 622 | 
            +
              const tokenRates = useTokenRates();
         | 
| 623 | 
            +
              return useMemo(() => ({
         | 
| 624 | 
            +
                chains,
         | 
| 625 | 
            +
                evmNetworks,
         | 
| 626 | 
            +
                tokens,
         | 
| 627 | 
            +
                tokenRates
         | 
| 628 | 
            +
              }), [chains, evmNetworks, tokens, tokenRates]);
         | 
| 629 | 
            +
            };
         | 
| 630 | 
            +
             | 
| 631 | 
            +
            function useBalances(addressesByToken) {
         | 
| 632 | 
            +
              // keep db data up to date
         | 
| 633 | 
            +
              useDbCacheBalancesSubscription();
         | 
| 634 | 
            +
              const balanceModules = useBalanceModules();
         | 
| 635 | 
            +
              const {
         | 
| 636 | 
            +
                balances
         | 
| 637 | 
            +
              } = useDbCache();
         | 
| 638 | 
            +
              const hydrate = useBalancesHydrate();
         | 
| 639 | 
            +
              return useMemo(() => new Balances(balances.filter(balance => {
         | 
| 208 640 | 
             
                // check that this balance is included in our queried balance modules
         | 
| 209 641 | 
             
                if (!balanceModules.map(({
         | 
| 210 642 | 
             
                  type
         | 
| @@ -221,136 +653,61 @@ balanceModules, chaindataProvider, addressesByToken, options = {}) { | |
| 221 653 |  | 
| 222 654 | 
             
                // keep this balance
         | 
| 223 655 | 
             
                return true;
         | 
| 224 | 
            -
              }) | 
| 656 | 
            +
              }),
         | 
| 225 657 | 
             
              // hydrate balance chains, evmNetworks, tokens and tokenRates
         | 
| 226 | 
            -
               | 
| 227 | 
            -
                chains,
         | 
| 228 | 
            -
                evmNetworks,
         | 
| 229 | 
            -
                tokens,
         | 
| 230 | 
            -
                tokenRates
         | 
| 231 | 
            -
              }), [balanceModules, addressesByToken, chains, evmNetworks, tokens, tokenRates]);
         | 
| 232 | 
            -
             | 
| 233 | 
            -
              // debounce every 100ms to prevent hammering UI with updates
         | 
| 234 | 
            -
              const [debouncedBalances, setDebouncedBalances] = useState(balances);
         | 
| 235 | 
            -
              useDebounce(() => balances && setDebouncedBalances(balances), 100, [balances]);
         | 
| 236 | 
            -
              return debouncedBalances;
         | 
| 658 | 
            +
              hydrate), [balances, hydrate, balanceModules, addressesByToken]);
         | 
| 237 659 | 
             
            }
         | 
| 238 660 |  | 
| 239 | 
            -
             | 
| 240 | 
            -
             | 
| 661 | 
            +
            /**
         | 
| 662 | 
            +
             * Given a collection of `Balances`, this hook returns a `BalancesStatus` summary for the collection.
         | 
| 663 | 
            +
             *
         | 
| 664 | 
            +
             * @param balances The collection of balances to get the status from.
         | 
| 665 | 
            +
             * @param isLoadingLocks Because the wallet currently fetches locks outside of the balances api, this param can be used to indicate that the locks are still loading, even if the `Balances` collection is not.
         | 
| 666 | 
            +
             * @returns An instance of `BalancesStatus` which represents the status of the balances collection.
         | 
| 241 667 |  | 
| 242 | 
            -
             | 
| 243 | 
            -
             | 
| 244 | 
            -
            //  | 
| 245 | 
            -
             | 
| 246 | 
            -
             | 
| 247 | 
            -
             | 
| 248 | 
            -
             | 
| 249 | 
            -
               | 
| 250 | 
            -
              // >({})
         | 
| 251 | 
            -
             | 
| 252 | 
            -
              const addSubscription = (key, balanceModule, chainConnectors, chaindataProvider, addressesByToken) => {
         | 
| 253 | 
            -
                // create subscription if it doesn't already exist
         | 
| 254 | 
            -
                if (!subscriptions[key] || subscriptions[key].refcount === 0) {
         | 
| 255 | 
            -
                  const generation = ((subscriptions[key]?.generation || 0) + 1) % Number.MAX_SAFE_INTEGER;
         | 
| 256 | 
            -
                  const unsub = balances(balanceModule, chainConnectors, chaindataProvider, addressesByToken, (error, balances) => {
         | 
| 257 | 
            -
                    if (error) return log.error(`Failed to fetch ${balanceModule.type} balances`, error);
         | 
| 258 | 
            -
                    if (!balances) return;
         | 
| 259 | 
            -
             | 
| 260 | 
            -
                    // ignore balances from old subscriptions which are still in the process of unsubscribing
         | 
| 261 | 
            -
                    if (subscriptions[key].generation !== generation) return;
         | 
| 262 | 
            -
                    const putBalances = Object.entries(balances.toJSON()).map(([id, balance]) => ({
         | 
| 263 | 
            -
                      id,
         | 
| 264 | 
            -
                      ...balance
         | 
| 265 | 
            -
                    }));
         | 
| 266 | 
            -
                    db.transaction("rw", db.balances, async () => await db.balances.bulkPut(putBalances));
         | 
| 267 | 
            -
                  });
         | 
| 268 | 
            -
                  subscriptions[key] = {
         | 
| 269 | 
            -
                    unsub,
         | 
| 270 | 
            -
                    refcount: 0,
         | 
| 271 | 
            -
                    generation
         | 
| 272 | 
            -
                  };
         | 
| 273 | 
            -
                }
         | 
| 668 | 
            +
             */
         | 
| 669 | 
            +
            const useBalancesStatus = (balances, isLoadingLocks) => useMemo(() => {
         | 
| 670 | 
            +
              // stale
         | 
| 671 | 
            +
              const staleChains = getStaleChains(balances);
         | 
| 672 | 
            +
              if (staleChains.length > 0) return {
         | 
| 673 | 
            +
                status: "stale",
         | 
| 674 | 
            +
                staleChains
         | 
| 675 | 
            +
              };
         | 
| 274 676 |  | 
| 275 | 
            -
             | 
| 276 | 
            -
             | 
| 677 | 
            +
              // fetching
         | 
| 678 | 
            +
              const hasCachedBalances = balances.each.some(b => b.status === "cache");
         | 
| 679 | 
            +
              if (hasCachedBalances || isLoadingLocks) return {
         | 
| 680 | 
            +
                status: "fetching"
         | 
| 277 681 | 
             
              };
         | 
| 278 | 
            -
             | 
| 279 | 
            -
             | 
| 280 | 
            -
             | 
| 281 | 
            -
             | 
| 282 | 
            -
                // drop the refcount by one
         | 
| 283 | 
            -
                subscriptions[key].refcount -= 1;
         | 
| 284 | 
            -
             | 
| 285 | 
            -
                // unsubscribe if refcount is now 0 (nobody wants this subcription anymore)
         | 
| 286 | 
            -
                if (subscriptions[key].refcount < 1) {
         | 
| 287 | 
            -
                  // remove subscription
         | 
| 288 | 
            -
                  subscriptions[key].unsub.then(unsub => unsub());
         | 
| 289 | 
            -
                  delete subscriptions[key];
         | 
| 290 | 
            -
             | 
| 291 | 
            -
                  // set this subscription's balances in the store to status: cache
         | 
| 292 | 
            -
                  db.transaction("rw", db.balances, async () => await db.balances.filter(balance => {
         | 
| 293 | 
            -
                    if (balance.source !== balanceModule.type) return false;
         | 
| 294 | 
            -
                    if (!Object.keys(addressesByToken).includes(balance.tokenId)) return false;
         | 
| 295 | 
            -
                    if (!addressesByToken[balance.tokenId].includes(balance.address)) return false;
         | 
| 296 | 
            -
                    return true;
         | 
| 297 | 
            -
                  }).modify({
         | 
| 298 | 
            -
                    status: "cache"
         | 
| 299 | 
            -
                  }));
         | 
| 300 | 
            -
                }
         | 
| 682 | 
            +
             | 
| 683 | 
            +
              // live
         | 
| 684 | 
            +
              return {
         | 
| 685 | 
            +
                status: "live"
         | 
| 301 686 | 
             
              };
         | 
| 302 | 
            -
             | 
| 303 | 
            -
             | 
| 304 | 
            -
              const tokens = useTokens(chaindataProvider);
         | 
| 305 | 
            -
              useEffect(() => {
         | 
| 306 | 
            -
                if (chainConnector === null) return;
         | 
| 307 | 
            -
                if (chainConnectorEvm === null) return;
         | 
| 308 | 
            -
                if (chaindataProvider === null) return;
         | 
| 309 | 
            -
                if (addressesByToken === null) return;
         | 
| 310 | 
            -
                const unsubs = balanceModules.map(balanceModule => {
         | 
| 311 | 
            -
                  const subscriptionKey = `${balanceModule.type}-${JSON.stringify(addressesByToken)}`;
         | 
| 312 | 
            -
             | 
| 313 | 
            -
                  // filter out tokens to only include those which this module knows how to fetch balances for
         | 
| 314 | 
            -
                  const moduleTokenIds = Object.values(tokens).filter(({
         | 
| 315 | 
            -
                    type
         | 
| 316 | 
            -
                  }) => type === balanceModule.type).map(({
         | 
| 317 | 
            -
                    id
         | 
| 318 | 
            -
                  }) => id);
         | 
| 319 | 
            -
                  const addressesByModuleToken = Object.fromEntries(Object.entries(addressesByToken).filter(([tokenId]) => moduleTokenIds.includes(tokenId)));
         | 
| 320 | 
            -
             | 
| 321 | 
            -
                  // add balance subscription for this module
         | 
| 322 | 
            -
                  addSubscription(subscriptionKey, balanceModule, {
         | 
| 323 | 
            -
                    substrate: chainConnector,
         | 
| 324 | 
            -
                    evm: chainConnectorEvm
         | 
| 325 | 
            -
                  }, chaindataProvider, addressesByModuleToken);
         | 
| 326 | 
            -
             | 
| 327 | 
            -
                  // return an unsub method, to be called when this effect unmounts
         | 
| 328 | 
            -
                  return () => removeSubscription(subscriptionKey, balanceModule, addressesByToken);
         | 
| 329 | 
            -
                });
         | 
| 330 | 
            -
                const unsubAll = () => unsubs.forEach(unsub => unsub());
         | 
| 331 | 
            -
                return unsubAll;
         | 
| 332 | 
            -
              }, [addressesByToken, balanceModules, chainConnector, chainConnectorEvm, chaindataProvider, tokens]);
         | 
| 333 | 
            -
            }
         | 
| 687 | 
            +
            }, [balances, isLoadingLocks]);
         | 
| 688 | 
            +
            const getStaleChains = balances => [...new Set(balances.sorted.filter(b => b.status === "stale").map(b => b.chain?.name ?? b.chainId ?? "Unknown"))];
         | 
| 334 689 |  | 
| 335 | 
            -
             | 
| 336 | 
            -
             | 
| 337 | 
            -
               | 
| 338 | 
            -
               | 
| 339 | 
            -
             | 
| 340 | 
            -
             | 
| 341 | 
            -
               | 
| 342 | 
            -
               | 
| 343 | 
            -
             | 
| 344 | 
            -
             | 
| 345 | 
            -
             | 
| 346 | 
            -
             | 
| 347 | 
            -
             | 
| 348 | 
            -
             | 
| 349 | 
            -
             | 
| 350 | 
            -
             | 
| 351 | 
            -
             | 
| 352 | 
            -
             | 
| 353 | 
            -
             | 
| 354 | 
            -
            }
         | 
| 690 | 
            +
            const BalancesProvider = ({
         | 
| 691 | 
            +
              balanceModules,
         | 
| 692 | 
            +
              onfinalityApiKey,
         | 
| 693 | 
            +
              withTestnets,
         | 
| 694 | 
            +
              children
         | 
| 695 | 
            +
            }) => /*#__PURE__*/jsx(WithTestnetsProvider, {
         | 
| 696 | 
            +
              withTestnets: withTestnets,
         | 
| 697 | 
            +
              children: /*#__PURE__*/jsx(ChaindataProvider, {
         | 
| 698 | 
            +
                onfinalityApiKey: onfinalityApiKey,
         | 
| 699 | 
            +
                children: /*#__PURE__*/jsx(ChainConnectorsProvider, {
         | 
| 700 | 
            +
                  onfinalityApiKey: onfinalityApiKey,
         | 
| 701 | 
            +
                  children: /*#__PURE__*/jsx(AllAddressesProvider, {
         | 
| 702 | 
            +
                    children: /*#__PURE__*/jsx(BalanceModulesProvider, {
         | 
| 703 | 
            +
                      balanceModules: balanceModules,
         | 
| 704 | 
            +
                      children: /*#__PURE__*/jsx(DbCacheProvider, {
         | 
| 705 | 
            +
                        children: children
         | 
| 706 | 
            +
                      })
         | 
| 707 | 
            +
                    })
         | 
| 708 | 
            +
                  })
         | 
| 709 | 
            +
                })
         | 
| 710 | 
            +
              })
         | 
| 711 | 
            +
            });
         | 
| 355 712 |  | 
| 356 | 
            -
            export { useBalances, useChain, useChaindata, useChains, useEvmNetwork, useEvmNetworks, useToken, useTokenRates, useTokens };
         | 
| 713 | 
            +
            export { AllAddressesProvider, BalanceModulesProvider, BalancesProvider, ChainConnectorsProvider, ChaindataProvider, DbCacheProvider, WithTestnetsProvider, createMulticastSubscription, getStaleChains, provideContext, useAllAddresses, useBalanceModules, useBalances, useBalancesHydrate, useBalancesStatus, useChain, useChainConnectors, useChaindata, useChains, useDbCache, useDbCacheBalancesSubscription, useDbCacheSubscription, useDbCacheTokenRatesSubscription, useEvmNetwork, useEvmNetworks, useMulticastSubscription, useToken, useTokenRate, useTokenRates, useTokens, useWithTestnets };
         |