2023-02-28 18:56:29 +00:00
|
|
|
import { LocalStorage } from '@vegaprotocol/utils';
|
2022-03-30 09:49:48 +00:00
|
|
|
import type { ReactNode } from 'react';
|
2022-10-03 18:12:34 +00:00
|
|
|
import { useCallback, useMemo, useRef, useState } from 'react';
|
2023-01-27 14:45:29 +00:00
|
|
|
import { WalletClientError } from '@vegaprotocol/wallet-client';
|
2022-10-03 18:12:34 +00:00
|
|
|
import type { VegaWalletContextShape } from '.';
|
|
|
|
import type {
|
|
|
|
PubKey,
|
|
|
|
Transaction,
|
|
|
|
VegaConnector,
|
|
|
|
} from './connectors/vega-connector';
|
2022-02-22 23:06:35 +00:00
|
|
|
import { VegaWalletContext } from './context';
|
2023-05-19 21:27:45 +00:00
|
|
|
import { WALLET_KEY, WALLET_RISK_ACCEPTED_KEY } from './storage';
|
2023-01-19 11:16:21 +00:00
|
|
|
import { ViewConnector } from './connectors';
|
2023-05-19 21:27:45 +00:00
|
|
|
import { useLocalStorage } from '@vegaprotocol/react-helpers';
|
2022-02-22 06:40:55 +00:00
|
|
|
|
2023-08-31 02:40:04 +00:00
|
|
|
type Networks =
|
|
|
|
| 'MAINNET'
|
|
|
|
| 'MAINNET_MIRROR'
|
|
|
|
| 'TESTNET'
|
|
|
|
| 'VALIDATOR_TESTNET'
|
|
|
|
| 'STAGNET1'
|
|
|
|
| 'DEVNET'
|
|
|
|
| 'CUSTOM';
|
|
|
|
|
|
|
|
interface VegaWalletLinks {
|
|
|
|
explorer: string;
|
|
|
|
concepts: string;
|
|
|
|
chromeExtensionUrl: string;
|
|
|
|
mozillaExtensionUrl: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface VegaWalletConfig {
|
|
|
|
network: Networks;
|
|
|
|
vegaUrl: string;
|
|
|
|
vegaWalletServiceUrl: string;
|
|
|
|
links: VegaWalletLinks;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ExternalLinks = {
|
|
|
|
VEGA_WALLET_URL_ABOUT: 'https://vega.xyz/wallet/#overview',
|
|
|
|
VEGA_WALLET_BROWSER_LIST: '',
|
|
|
|
};
|
|
|
|
|
2022-02-22 06:40:55 +00:00
|
|
|
interface VegaWalletProviderProps {
|
|
|
|
children: ReactNode;
|
2023-08-31 02:40:04 +00:00
|
|
|
config: VegaWalletConfig;
|
2022-02-22 06:40:55 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 02:40:04 +00:00
|
|
|
export const VegaWalletProvider = ({
|
|
|
|
children,
|
|
|
|
config,
|
|
|
|
}: VegaWalletProviderProps) => {
|
2022-10-03 18:12:34 +00:00
|
|
|
// Current selected pubKey
|
|
|
|
const [pubKey, setPubKey] = useState<string | null>(null);
|
2023-01-19 11:16:21 +00:00
|
|
|
const [isReadOnly, setIsReadOnly] = useState<boolean>(false);
|
2022-02-24 00:03:59 +00:00
|
|
|
|
2023-04-20 11:56:48 +00:00
|
|
|
// Array of public keys retrieved from the connector
|
2022-10-03 18:12:34 +00:00
|
|
|
const [pubKeys, setPubKeys] = useState<PubKey[] | null>(null);
|
2022-02-24 00:03:59 +00:00
|
|
|
|
|
|
|
// Reference to the current connector instance
|
2022-02-22 06:40:55 +00:00
|
|
|
const connector = useRef<VegaConnector | null>(null);
|
|
|
|
|
2022-10-03 18:12:34 +00:00
|
|
|
const selectPubKey = useCallback((pk: string) => {
|
|
|
|
setPubKey(pk);
|
|
|
|
LocalStorage.setItem(WALLET_KEY, pk);
|
|
|
|
}, []);
|
2022-03-20 01:31:44 +00:00
|
|
|
|
2023-04-20 11:56:48 +00:00
|
|
|
const fetchPubKeys = useCallback(async () => {
|
|
|
|
if (!connector.current) {
|
|
|
|
throw new Error('No connector');
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
const keys = await connector.current.connect();
|
|
|
|
|
|
|
|
if (keys?.length) {
|
|
|
|
setPubKeys(keys);
|
|
|
|
setIsReadOnly(connector.current instanceof ViewConnector);
|
|
|
|
return keys;
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
if (err instanceof WalletClientError) {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
|
2022-10-03 18:12:34 +00:00
|
|
|
const connect = useCallback(async (c: VegaConnector) => {
|
|
|
|
connector.current = c;
|
|
|
|
try {
|
|
|
|
const keys = await connector.current.connect();
|
2022-03-20 01:31:44 +00:00
|
|
|
|
2022-10-03 18:12:34 +00:00
|
|
|
if (keys?.length) {
|
|
|
|
setPubKeys(keys);
|
2023-01-19 11:16:21 +00:00
|
|
|
setIsReadOnly(connector.current instanceof ViewConnector);
|
2022-10-03 18:12:34 +00:00
|
|
|
const lastUsedPubKey = LocalStorage.getItem(WALLET_KEY);
|
|
|
|
const foundKey = keys.find((key) => key.publicKey === lastUsedPubKey);
|
|
|
|
if (foundKey) {
|
|
|
|
setPubKey(foundKey.publicKey);
|
|
|
|
} else {
|
|
|
|
setPubKey(keys[0].publicKey);
|
2022-03-20 01:31:44 +00:00
|
|
|
}
|
2022-02-23 19:24:30 +00:00
|
|
|
|
2022-10-03 18:12:34 +00:00
|
|
|
return keys;
|
|
|
|
} else {
|
2022-03-08 18:57:07 +00:00
|
|
|
return null;
|
2022-02-23 19:24:30 +00:00
|
|
|
}
|
2022-10-03 18:12:34 +00:00
|
|
|
} catch (err) {
|
2023-01-27 14:45:29 +00:00
|
|
|
if (err instanceof WalletClientError) {
|
2022-12-05 16:51:53 +00:00
|
|
|
throw err;
|
|
|
|
}
|
2022-10-03 18:12:34 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}, []);
|
2022-02-22 06:40:55 +00:00
|
|
|
|
|
|
|
const disconnect = useCallback(async () => {
|
2022-12-30 18:05:27 +00:00
|
|
|
// always clear state after attempted disconnection.. this
|
|
|
|
// is because long lived token sessions (used in tests)
|
|
|
|
// cannot be cleared. Clearing state will force user to reconnect
|
|
|
|
// again as expected
|
|
|
|
setPubKeys(null);
|
|
|
|
setPubKey(null);
|
2023-01-19 11:16:21 +00:00
|
|
|
setIsReadOnly(false);
|
2022-12-30 18:05:27 +00:00
|
|
|
LocalStorage.removeItem(WALLET_KEY);
|
2022-02-22 06:40:55 +00:00
|
|
|
try {
|
2022-02-22 23:06:35 +00:00
|
|
|
await connector.current?.disconnect();
|
2022-02-22 06:40:55 +00:00
|
|
|
connector.current = null;
|
|
|
|
} catch (err) {
|
2022-02-23 05:39:12 +00:00
|
|
|
console.error(err);
|
2022-12-30 18:05:27 +00:00
|
|
|
connector.current = null;
|
2022-02-22 06:40:55 +00:00
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
|
2022-10-03 18:12:34 +00:00
|
|
|
const sendTx = useCallback((pubkey: string, transaction: Transaction) => {
|
2022-05-17 13:04:41 +00:00
|
|
|
if (!connector.current) {
|
2022-10-03 18:12:34 +00:00
|
|
|
throw new Error('No connector');
|
2022-05-17 13:04:41 +00:00
|
|
|
}
|
2022-10-03 18:12:34 +00:00
|
|
|
return connector.current.sendTx(pubkey, transaction);
|
2022-05-17 13:04:41 +00:00
|
|
|
}, []);
|
2022-03-10 00:39:59 +00:00
|
|
|
|
2023-05-19 21:27:45 +00:00
|
|
|
const [riskAcceptedValue] = useLocalStorage(WALLET_RISK_ACCEPTED_KEY);
|
|
|
|
const acknowledgeNeeded =
|
2023-08-31 02:40:04 +00:00
|
|
|
config.network === 'MAINNET' && riskAcceptedValue !== 'true';
|
2023-05-19 21:27:45 +00:00
|
|
|
|
2022-02-23 05:05:39 +00:00
|
|
|
const contextValue = useMemo<VegaWalletContextShape>(() => {
|
2022-02-22 06:40:55 +00:00
|
|
|
return {
|
2023-08-31 02:40:04 +00:00
|
|
|
vegaUrl: config.vegaUrl,
|
|
|
|
vegaWalletServiceUrl: config.vegaWalletServiceUrl,
|
|
|
|
network: config.network,
|
|
|
|
links: {
|
|
|
|
explorer: config.links.explorer,
|
|
|
|
about: ExternalLinks.VEGA_WALLET_URL_ABOUT,
|
|
|
|
browserList: ExternalLinks.VEGA_WALLET_BROWSER_LIST,
|
|
|
|
concepts: config.links.concepts,
|
|
|
|
chromeExtensionUrl: config.links.chromeExtensionUrl,
|
|
|
|
mozillaExtensionUrl: config.links.mozillaExtensionUrl,
|
|
|
|
},
|
2023-01-19 11:16:21 +00:00
|
|
|
isReadOnly,
|
2022-10-03 18:12:34 +00:00
|
|
|
pubKey,
|
|
|
|
pubKeys,
|
|
|
|
selectPubKey,
|
2022-02-22 06:40:55 +00:00
|
|
|
connect,
|
|
|
|
disconnect,
|
2022-03-10 00:39:59 +00:00
|
|
|
sendTx,
|
2023-04-20 11:56:48 +00:00
|
|
|
fetchPubKeys,
|
2023-05-19 21:27:45 +00:00
|
|
|
acknowledgeNeeded,
|
2022-10-03 18:12:34 +00:00
|
|
|
};
|
2023-04-20 11:56:48 +00:00
|
|
|
}, [
|
2023-08-31 02:40:04 +00:00
|
|
|
config,
|
2023-04-20 11:56:48 +00:00
|
|
|
isReadOnly,
|
|
|
|
pubKey,
|
|
|
|
pubKeys,
|
|
|
|
selectPubKey,
|
|
|
|
connect,
|
|
|
|
disconnect,
|
|
|
|
sendTx,
|
|
|
|
fetchPubKeys,
|
2023-05-19 21:27:45 +00:00
|
|
|
acknowledgeNeeded,
|
2023-04-20 11:56:48 +00:00
|
|
|
]);
|
2022-02-22 06:40:55 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<VegaWalletContext.Provider value={contextValue}>
|
|
|
|
{children}
|
|
|
|
</VegaWalletContext.Provider>
|
|
|
|
);
|
|
|
|
};
|