feat: make blockchain JSON RPC configurable from settings (#494)
* feat: make blockchain JSON RPC configurable from settings * chore: expose the settings directly * feat: restart bee node when blockchain RPC changes, add notification and error logging
This commit is contained in:
@@ -13,6 +13,7 @@ import { Loading } from '../../components/Loading'
|
|||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
import { Context as TopUpContext } from '../../providers/TopUp'
|
||||||
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { createGiftWallet } from '../../utils/desktop'
|
import { createGiftWallet } from '../../utils/desktop'
|
||||||
import { ResolvedWallet } from '../../utils/wallet'
|
import { ResolvedWallet } from '../../utils/wallet'
|
||||||
import { Token } from '../../models/Token'
|
import { Token } from '../../models/Token'
|
||||||
@@ -21,7 +22,8 @@ const GIFT_WALLET_FUND_DAI_AMOUNT = Token.fromDecimal('0.1', 18)
|
|||||||
const GIFT_WALLET_FUND_BZZ_AMOUNT = Token.fromDecimal('0.5', 16)
|
const GIFT_WALLET_FUND_BZZ_AMOUNT = Token.fromDecimal('0.5', 16)
|
||||||
|
|
||||||
export default function Index(): ReactElement {
|
export default function Index(): ReactElement {
|
||||||
const { giftWallets, addGiftWallet, provider } = useContext(TopUpContext)
|
const { giftWallets, addGiftWallet } = useContext(TopUpContext)
|
||||||
|
const { provider } = useContext(SettingsContext)
|
||||||
const { balance } = useContext(BeeContext)
|
const { balance } = useContext(BeeContext)
|
||||||
|
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
|
|||||||
@@ -2,11 +2,27 @@ import CircularProgress from '@material-ui/core/CircularProgress'
|
|||||||
import { ReactElement, useContext } from 'react'
|
import { ReactElement, useContext } from 'react'
|
||||||
import ExpandableList from '../../components/ExpandableList'
|
import ExpandableList from '../../components/ExpandableList'
|
||||||
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
import ExpandableListItemInput from '../../components/ExpandableListItemInput'
|
||||||
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as SettingsContext } from '../../providers/Settings'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
|
import config from '../../config'
|
||||||
|
import { useSnackbar } from 'notistack'
|
||||||
|
|
||||||
export default function SettingsPage(): ReactElement {
|
export default function SettingsPage(): ReactElement {
|
||||||
const { apiUrl, apiDebugUrl, setApiUrl, setDebugApiUrl, lockedApiSettings, config, isLoading } =
|
const {
|
||||||
useContext(SettingsContext)
|
apiUrl,
|
||||||
|
apiDebugUrl,
|
||||||
|
setApiUrl,
|
||||||
|
setDebugApiUrl,
|
||||||
|
lockedApiSettings,
|
||||||
|
cors,
|
||||||
|
dataDir,
|
||||||
|
ensResolver,
|
||||||
|
providerUrl,
|
||||||
|
isLoading,
|
||||||
|
setAndPersistJsonRpcProvider,
|
||||||
|
} = useContext(SettingsContext)
|
||||||
|
const { refresh } = useContext(BeeContext)
|
||||||
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
if (isLoading) {
|
if (isLoading) {
|
||||||
return (
|
return (
|
||||||
@@ -16,31 +32,46 @@ export default function SettingsPage(): ReactElement {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Run within Bee Desktop, display read only config
|
|
||||||
if (config) {
|
|
||||||
return (
|
|
||||||
<ExpandableList label="Bee Desktop Settings" defaultOpen>
|
|
||||||
<ExpandableListItemInput label="Bee API" value={config['api-addr']} locked />
|
|
||||||
<ExpandableListItemInput label="Bee Debug API" value={config['debug-api-addr']} locked />
|
|
||||||
<ExpandableListItemInput label="CORS" value={config['cors-allowed-origins']} locked />
|
|
||||||
<ExpandableListItemInput label="Data DIR" value={config['data-dir']} locked />
|
|
||||||
<ExpandableListItemInput label="ENS resolver URL" value={config['resolver-options']} locked />
|
|
||||||
{config['swap-endpoint'] && (
|
|
||||||
<ExpandableListItemInput label="SWAP endpoint" value={config['swap-endpoint']} locked />
|
|
||||||
)}
|
|
||||||
</ExpandableList>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ExpandableList label="API Settings" defaultOpen>
|
<>
|
||||||
<ExpandableListItemInput label="Bee API" value={apiUrl} onConfirm={setApiUrl} locked={lockedApiSettings} />
|
<ExpandableList label="API Settings" defaultOpen>
|
||||||
<ExpandableListItemInput
|
<ExpandableListItemInput
|
||||||
label="Bee Debug API"
|
label="Bee API"
|
||||||
value={apiDebugUrl}
|
value={apiUrl}
|
||||||
onConfirm={setDebugApiUrl}
|
onConfirm={setApiUrl}
|
||||||
locked={lockedApiSettings}
|
locked={lockedApiSettings || config.BEE_DESKTOP_ENABLED}
|
||||||
/>
|
/>
|
||||||
</ExpandableList>
|
<ExpandableListItemInput
|
||||||
|
label="Bee Debug API"
|
||||||
|
value={apiDebugUrl}
|
||||||
|
onConfirm={setDebugApiUrl}
|
||||||
|
locked={lockedApiSettings || config.BEE_DESKTOP_ENABLED}
|
||||||
|
/>
|
||||||
|
<ExpandableListItemInput
|
||||||
|
label="Blockchain RPC URL"
|
||||||
|
value={providerUrl}
|
||||||
|
helperText="Changing the value will restart your bee node."
|
||||||
|
confirmLabel="Save and restart"
|
||||||
|
onConfirm={value => {
|
||||||
|
setAndPersistJsonRpcProvider(value)
|
||||||
|
.then(() => {
|
||||||
|
refresh()
|
||||||
|
enqueueSnackbar('Settings changed, restarting bee node...', { variant: 'success' })
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.log(error) //eslint-disable-line
|
||||||
|
enqueueSnackbar(`Failed to change RPC endpoint. Error: ${error}`, { variant: 'success' })
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</ExpandableList>
|
||||||
|
{config.BEE_DESKTOP_ENABLED && (
|
||||||
|
<ExpandableList label="Desktop Settings" defaultOpen>
|
||||||
|
<ExpandableListItemInput label="CORS" value={cors ?? '-'} locked />
|
||||||
|
<ExpandableListItemInput label="Data DIR" value={dataDir ?? '-'} locked />
|
||||||
|
<ExpandableListItemInput label="ENS resolver URL" value={ensResolver ?? '-'} locked />
|
||||||
|
</ExpandableList>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import { ProgressIndicator } from '../../components/ProgressIndicator'
|
|||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { SwarmDivider } from '../../components/SwarmDivider'
|
import { SwarmDivider } from '../../components/SwarmDivider'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { sleepMs } from '../../utils'
|
import { sleepMs } from '../../utils'
|
||||||
import { restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
import { restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
||||||
@@ -23,7 +23,7 @@ import { BeeModes } from '@ethersphere/bee-js'
|
|||||||
export function GiftCardFund(): ReactElement {
|
export function GiftCardFund(): ReactElement {
|
||||||
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
||||||
const { nodeAddresses, balance, nodeInfo } = useContext(BeeContext)
|
const { nodeAddresses, balance, nodeInfo } = useContext(BeeContext)
|
||||||
const { provider, providerUrl } = useContext(TopUpContext)
|
const { provider, providerUrl } = useContext(SettingsContext)
|
||||||
|
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const [wallet, setWallet] = useState<ResolvedWallet | null>(null)
|
const [wallet, setWallet] = useState<ResolvedWallet | null>(null)
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { useSnackbar } from 'notistack'
|
|||||||
import { ReactElement, useContext, useState } from 'react'
|
import { ReactElement, useContext, useState } from 'react'
|
||||||
import ArrowRight from 'remixicon-react/ArrowRightLineIcon'
|
import ArrowRight from 'remixicon-react/ArrowRightLineIcon'
|
||||||
import { useNavigate } from 'react-router'
|
import { useNavigate } from 'react-router'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { HistoryHeader } from '../../components/HistoryHeader'
|
import { HistoryHeader } from '../../components/HistoryHeader'
|
||||||
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
import { ProgressIndicator } from '../../components/ProgressIndicator'
|
||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
@@ -16,7 +16,7 @@ import { ROUTES } from '../../routes'
|
|||||||
import { Rpc } from '../../utils/rpc'
|
import { Rpc } from '../../utils/rpc'
|
||||||
|
|
||||||
export function GiftCardTopUpIndex(): ReactElement {
|
export function GiftCardTopUpIndex(): ReactElement {
|
||||||
const { provider } = useContext(TopUpContext)
|
const { provider } = useContext(SettingsContext)
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const [giftCode, setGiftCode] = useState('')
|
const [giftCode, setGiftCode] = useState('')
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ import { SwarmTextInput } from '../../components/SwarmTextInput'
|
|||||||
import { BzzToken } from '../../models/BzzToken'
|
import { BzzToken } from '../../models/BzzToken'
|
||||||
import { DaiToken } from '../../models/DaiToken'
|
import { DaiToken } from '../../models/DaiToken'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { sleepMs } from '../../utils'
|
import { sleepMs } from '../../utils'
|
||||||
import { getBzzPriceAsDai, performSwap, restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
import { getBzzPriceAsDai, performSwap, restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
||||||
@@ -37,7 +37,7 @@ export function Swap({ header }: Props): ReactElement {
|
|||||||
const [userInputSwap, setUserInputSwap] = useState<string | null>(null)
|
const [userInputSwap, setUserInputSwap] = useState<string | null>(null)
|
||||||
const [price, setPrice] = useState(DaiToken.fromDecimal('0.6', 18))
|
const [price, setPrice] = useState(DaiToken.fromDecimal('0.6', 18))
|
||||||
|
|
||||||
const { providerUrl } = useContext(TopUpContext)
|
const { providerUrl } = useContext(SettingsContext)
|
||||||
const { balance, nodeAddresses, nodeInfo } = useContext(BeeContext)
|
const { balance, nodeAddresses, nodeInfo } = useContext(BeeContext)
|
||||||
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import { HistoryHeader } from '../../components/HistoryHeader'
|
|||||||
import { SwarmButton } from '../../components/SwarmButton'
|
import { SwarmButton } from '../../components/SwarmButton'
|
||||||
import { ROUTES } from '../../routes'
|
import { ROUTES } from '../../routes'
|
||||||
import { Context as BeeContext } from '../../providers/Bee'
|
import { Context as BeeContext } from '../../providers/Bee'
|
||||||
import { Context as TopUpContext } from '../../providers/TopUp'
|
import { Context as SettingsContext } from '../../providers/Settings'
|
||||||
import config from '../../config'
|
import config from '../../config'
|
||||||
import { BeeModes } from '@ethersphere/bee-js'
|
import { BeeModes } from '@ethersphere/bee-js'
|
||||||
import { restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
import { restartBeeNode, upgradeToLightNode } from '../../utils/desktop'
|
||||||
@@ -40,7 +40,7 @@ export default function TopUp(): ReactElement {
|
|||||||
const styles = useStyles()
|
const styles = useStyles()
|
||||||
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
const isBeeDesktop = config.BEE_DESKTOP_ENABLED
|
||||||
const { balance, nodeInfo } = useContext(BeeContext)
|
const { balance, nodeInfo } = useContext(BeeContext)
|
||||||
const { providerUrl } = useContext(TopUpContext)
|
const { providerUrl } = useContext(SettingsContext)
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const { enqueueSnackbar } = useSnackbar()
|
const { enqueueSnackbar } = useSnackbar()
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ import { Token } from '../models/Token'
|
|||||||
import type { Balance, ChequebookBalance, Settlements } from '../types'
|
import type { Balance, ChequebookBalance, Settlements } from '../types'
|
||||||
import { WalletAddress } from '../utils/wallet'
|
import { WalletAddress } from '../utils/wallet'
|
||||||
import { Context as SettingsContext } from './Settings'
|
import { Context as SettingsContext } from './Settings'
|
||||||
import { Context as TopUpContext } from './TopUp'
|
|
||||||
|
|
||||||
const REFRESH_WHEN_OK = 30_000
|
const REFRESH_WHEN_OK = 30_000
|
||||||
const REFRESH_WHEN_ERROR = 5_000
|
const REFRESH_WHEN_ERROR = 5_000
|
||||||
@@ -192,8 +191,7 @@ function getStatus(
|
|||||||
let isRefreshing = false
|
let isRefreshing = false
|
||||||
|
|
||||||
export function Provider({ children }: Props): ReactElement {
|
export function Provider({ children }: Props): ReactElement {
|
||||||
const { beeApi, beeDebugApi } = useContext(SettingsContext)
|
const { beeApi, beeDebugApi, provider } = useContext(SettingsContext)
|
||||||
const { provider } = useContext(TopUpContext)
|
|
||||||
const [apiHealth, setApiHealth] = useState<boolean>(false)
|
const [apiHealth, setApiHealth] = useState<boolean>(false)
|
||||||
const [debugApiHealth, setDebugApiHealth] = useState<Health | null>(null)
|
const [debugApiHealth, setDebugApiHealth] = useState<Health | null>(null)
|
||||||
const [nodeAddresses, setNodeAddresses] = useState<NodeAddresses | null>(null)
|
const [nodeAddresses, setNodeAddresses] = useState<NodeAddresses | null>(null)
|
||||||
|
|||||||
+50
-11
@@ -1,32 +1,50 @@
|
|||||||
import { Bee, BeeDebug } from '@ethersphere/bee-js'
|
import { Bee, BeeDebug } from '@ethersphere/bee-js'
|
||||||
|
import { providers } from 'ethers'
|
||||||
import { createContext, ReactChild, ReactElement, useEffect, useState } from 'react'
|
import { createContext, ReactChild, ReactElement, useEffect, useState } from 'react'
|
||||||
import { config } from '../config'
|
import { config as appConfig } from '../config'
|
||||||
import { BeeConfig, useGetBeeConfig } from '../hooks/apiHooks'
|
import { useGetBeeConfig } from '../hooks/apiHooks'
|
||||||
|
import { restartBeeNode, setJsonRpcInDesktop } from '../utils/desktop'
|
||||||
|
|
||||||
|
const LocalStorageKeys = {
|
||||||
|
providerUrl: 'json-rpc-provider',
|
||||||
|
}
|
||||||
|
|
||||||
|
const providerUrl = localStorage.getItem('json-rpc-provider') || appConfig.DEFAULT_RPC_URL
|
||||||
|
|
||||||
interface ContextInterface {
|
interface ContextInterface {
|
||||||
apiUrl: string
|
apiUrl: string
|
||||||
apiDebugUrl: string
|
apiDebugUrl: string
|
||||||
beeApi: Bee | null
|
beeApi: Bee | null
|
||||||
beeDebugApi: BeeDebug | null
|
beeDebugApi: BeeDebug | null
|
||||||
setApiUrl: (url: string) => void
|
|
||||||
setDebugApiUrl: (url: string) => void
|
|
||||||
lockedApiSettings: boolean
|
lockedApiSettings: boolean
|
||||||
desktopApiKey: string
|
desktopApiKey: string
|
||||||
config: BeeConfig | null
|
providerUrl: string
|
||||||
|
provider: providers.JsonRpcProvider
|
||||||
|
cors: string | null
|
||||||
|
dataDir: string | null
|
||||||
|
ensResolver: string | null
|
||||||
|
setApiUrl: (url: string) => void
|
||||||
|
setDebugApiUrl: (url: string) => void
|
||||||
|
setAndPersistJsonRpcProvider: (url: string) => Promise<void>
|
||||||
isLoading: boolean
|
isLoading: boolean
|
||||||
error: Error | null
|
error: Error | null
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialValues: ContextInterface = {
|
const initialValues: ContextInterface = {
|
||||||
apiUrl: config.BEE_API_HOST,
|
apiUrl: appConfig.BEE_API_HOST,
|
||||||
apiDebugUrl: config.BEE_DEBUG_API_HOST,
|
apiDebugUrl: appConfig.BEE_DEBUG_API_HOST,
|
||||||
beeApi: null,
|
beeApi: null,
|
||||||
beeDebugApi: null,
|
beeDebugApi: null,
|
||||||
setApiUrl: () => {}, // eslint-disable-line
|
setApiUrl: () => {}, // eslint-disable-line
|
||||||
setDebugApiUrl: () => {}, // eslint-disable-line
|
setDebugApiUrl: () => {}, // eslint-disable-line
|
||||||
lockedApiSettings: false,
|
lockedApiSettings: false,
|
||||||
desktopApiKey: '',
|
desktopApiKey: '',
|
||||||
config: null,
|
setAndPersistJsonRpcProvider: async () => {}, // eslint-disable-line
|
||||||
|
providerUrl,
|
||||||
|
provider: new providers.JsonRpcProvider(providerUrl),
|
||||||
|
cors: null,
|
||||||
|
dataDir: null,
|
||||||
|
ensResolver: null,
|
||||||
isLoading: true,
|
isLoading: true,
|
||||||
error: null,
|
error: null,
|
||||||
}
|
}
|
||||||
@@ -51,10 +69,26 @@ export function Provider({
|
|||||||
const [apiDebugUrl, setDebugApiUrl] = useState<string>(initialValues.apiDebugUrl)
|
const [apiDebugUrl, setDebugApiUrl] = useState<string>(initialValues.apiDebugUrl)
|
||||||
const [beeApi, setBeeApi] = useState<Bee | null>(null)
|
const [beeApi, setBeeApi] = useState<Bee | null>(null)
|
||||||
const [beeDebugApi, setBeeDebugApi] = useState<BeeDebug | null>(null)
|
const [beeDebugApi, setBeeDebugApi] = useState<BeeDebug | null>(null)
|
||||||
const [lockedApiSettings] = useState<boolean>(Boolean(extLockedApiSettings))
|
|
||||||
const [desktopApiKey, setDesktopApiKey] = useState<string>(initialValues.desktopApiKey)
|
const [desktopApiKey, setDesktopApiKey] = useState<string>(initialValues.desktopApiKey)
|
||||||
|
const [providerUrl, setProviderUrl] = useState(initialValues.providerUrl)
|
||||||
|
const [provider, setProvider] = useState(initialValues.provider)
|
||||||
const { config, isLoading, error } = useGetBeeConfig()
|
const { config, isLoading, error } = useGetBeeConfig()
|
||||||
|
|
||||||
|
async function setAndPersistJsonRpcProvider(providerUrl: string) {
|
||||||
|
try {
|
||||||
|
localStorage.setItem(LocalStorageKeys.providerUrl, providerUrl)
|
||||||
|
setProviderUrl(providerUrl)
|
||||||
|
setProvider(new providers.JsonRpcProvider(providerUrl))
|
||||||
|
|
||||||
|
if (appConfig.BEE_DESKTOP_ENABLED) {
|
||||||
|
await setJsonRpcInDesktop(providerUrl)
|
||||||
|
await restartBeeNode()
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error) // eslint-disable-line
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function makeHttpUrl(string: string): string {
|
function makeHttpUrl(string: string): string {
|
||||||
if (!string.startsWith('http')) {
|
if (!string.startsWith('http')) {
|
||||||
return `http://${string}`
|
return `http://${string}`
|
||||||
@@ -104,9 +138,14 @@ export function Provider({
|
|||||||
beeDebugApi,
|
beeDebugApi,
|
||||||
setApiUrl,
|
setApiUrl,
|
||||||
setDebugApiUrl,
|
setDebugApiUrl,
|
||||||
lockedApiSettings,
|
lockedApiSettings: Boolean(extLockedApiSettings),
|
||||||
desktopApiKey,
|
desktopApiKey,
|
||||||
config,
|
provider,
|
||||||
|
providerUrl,
|
||||||
|
cors: config?.['cors-allowed-origins'] ?? null,
|
||||||
|
dataDir: config?.['data-dir'] ?? null,
|
||||||
|
ensResolver: config?.['resolver-options'] ?? null,
|
||||||
|
setAndPersistJsonRpcProvider,
|
||||||
isLoading,
|
isLoading,
|
||||||
error,
|
error,
|
||||||
}}
|
}}
|
||||||
|
|||||||
+4
-26
@@ -1,30 +1,20 @@
|
|||||||
import { providers, Wallet } from 'ethers'
|
import { Wallet } from 'ethers'
|
||||||
import { createContext, ReactElement, useEffect, useState } from 'react'
|
import { createContext, ReactElement, useContext, useEffect, useState } from 'react'
|
||||||
import config from '../config'
|
import { Context as SettingsContext } from './Settings'
|
||||||
import { setJsonRpcInDesktop } from '../utils/desktop'
|
|
||||||
|
|
||||||
const LocalStorageKeys = {
|
const LocalStorageKeys = {
|
||||||
providerUrl: 'json-rpc-provider',
|
|
||||||
depositWallet: 'deposit-wallet',
|
depositWallet: 'deposit-wallet',
|
||||||
giftWallets: 'gift-wallets',
|
giftWallets: 'gift-wallets',
|
||||||
invitation: 'invitation',
|
invitation: 'invitation',
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ContextInterface {
|
interface ContextInterface {
|
||||||
providerUrl: string
|
|
||||||
provider: providers.JsonRpcProvider
|
|
||||||
giftWallets: Wallet[]
|
giftWallets: Wallet[]
|
||||||
setProviderUrl: (providerUrl: string) => void
|
|
||||||
addGiftWallet: (wallet: Wallet) => void
|
addGiftWallet: (wallet: Wallet) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
const providerUrl = localStorage.getItem('json-rpc-provider') || config.DEFAULT_RPC_URL
|
|
||||||
|
|
||||||
const initialValues: ContextInterface = {
|
const initialValues: ContextInterface = {
|
||||||
providerUrl,
|
|
||||||
provider: new providers.JsonRpcProvider(providerUrl),
|
|
||||||
giftWallets: [],
|
giftWallets: [],
|
||||||
setProviderUrl: () => {}, // eslint-disable-line
|
|
||||||
addGiftWallet: () => {}, // eslint-disable-line
|
addGiftWallet: () => {}, // eslint-disable-line
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -36,9 +26,8 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function Provider({ children }: Props): ReactElement {
|
export function Provider({ children }: Props): ReactElement {
|
||||||
const [providerUrl, setProviderUrl] = useState(initialValues.providerUrl)
|
|
||||||
const [provider, setProvider] = useState(initialValues.provider)
|
|
||||||
const [giftWallets, setGiftWallets] = useState(initialValues.giftWallets)
|
const [giftWallets, setGiftWallets] = useState(initialValues.giftWallets)
|
||||||
|
const { provider } = useContext(SettingsContext)
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (provider === null) return
|
if (provider === null) return
|
||||||
@@ -50,14 +39,6 @@ export function Provider({ children }: Props): ReactElement {
|
|||||||
}
|
}
|
||||||
}, [provider])
|
}, [provider])
|
||||||
|
|
||||||
function setAndPersistJsonRpcProvider(providerUrl: string) {
|
|
||||||
localStorage.setItem(LocalStorageKeys.providerUrl, providerUrl)
|
|
||||||
setProviderUrl(providerUrl)
|
|
||||||
setProvider(new providers.JsonRpcProvider(providerUrl))
|
|
||||||
// eslint-disable-next-line no-console
|
|
||||||
setJsonRpcInDesktop(providerUrl).catch(console.error)
|
|
||||||
}
|
|
||||||
|
|
||||||
function addGiftWallet(wallet: Wallet) {
|
function addGiftWallet(wallet: Wallet) {
|
||||||
const newArray = [...giftWallets, wallet]
|
const newArray = [...giftWallets, wallet]
|
||||||
localStorage.setItem(LocalStorageKeys.giftWallets, JSON.stringify(newArray.map(x => x.privateKey)))
|
localStorage.setItem(LocalStorageKeys.giftWallets, JSON.stringify(newArray.map(x => x.privateKey)))
|
||||||
@@ -67,10 +48,7 @@ export function Provider({ children }: Props): ReactElement {
|
|||||||
return (
|
return (
|
||||||
<Context.Provider
|
<Context.Provider
|
||||||
value={{
|
value={{
|
||||||
providerUrl,
|
|
||||||
provider,
|
|
||||||
giftWallets,
|
giftWallets,
|
||||||
setProviderUrl: setAndPersistJsonRpcProvider,
|
|
||||||
addGiftWallet,
|
addGiftWallet,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
|||||||
+2
-2
@@ -6,8 +6,8 @@ export function getJson<T extends Record<string, any>>(url: string): Promise<T>
|
|||||||
return sendRequest(url, 'GET') as Promise<T>
|
return sendRequest(url, 'GET') as Promise<T>
|
||||||
}
|
}
|
||||||
|
|
||||||
export function postJson(url: string, data?: Record<string, any>): Promise<Record<string, unknown>> {
|
export function postJson<T extends Record<string, any>>(url: string, data?: T): Promise<T> {
|
||||||
return sendRequest(url, 'POST', data)
|
return sendRequest(url, 'POST', data) as Promise<T>
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function sendRequest(
|
export async function sendRequest(
|
||||||
|
|||||||
Reference in New Issue
Block a user