2022-06-21 23:20:53 +00:00
|
|
|
import type { Dispatch, SetStateAction } from 'react';
|
|
|
|
import { useState, useEffect } from 'react';
|
|
|
|
import { LocalStorage } from '@vegaprotocol/react-helpers';
|
2022-07-12 16:34:54 +00:00
|
|
|
import type {
|
|
|
|
Environment,
|
|
|
|
Configuration,
|
|
|
|
ConfigStatus,
|
|
|
|
Networks,
|
|
|
|
} from '../types';
|
2022-06-21 23:20:53 +00:00
|
|
|
import { validateConfiguration } from '../utils/validate-configuration';
|
|
|
|
import { promiseRaceToSuccess } from '../utils/promise-race-success';
|
|
|
|
|
|
|
|
export const LOCAL_STORAGE_NETWORK_KEY = 'vegaNetworkConfig';
|
|
|
|
|
|
|
|
export type EnvironmentWithOptionalUrl = Partial<Environment> &
|
|
|
|
Omit<Environment, 'VEGA_URL'>;
|
|
|
|
|
|
|
|
const requestToNode = async (url: string, index: number): Promise<number> => {
|
|
|
|
const response = await fetch(url);
|
|
|
|
if (!response.ok) {
|
|
|
|
throw new Error(`Failed connecting to node: ${url}.`);
|
|
|
|
}
|
|
|
|
return index;
|
|
|
|
};
|
|
|
|
|
2022-07-12 16:34:54 +00:00
|
|
|
const getCacheKey = (env: Networks) => `${LOCAL_STORAGE_NETWORK_KEY}-${env}`;
|
|
|
|
|
|
|
|
const getCachedConfig = (env: Networks) => {
|
|
|
|
const key = getCacheKey(env);
|
|
|
|
const value = LocalStorage.getItem(key);
|
2022-06-21 23:20:53 +00:00
|
|
|
|
|
|
|
if (value) {
|
|
|
|
try {
|
|
|
|
const config = JSON.parse(value) as Configuration;
|
|
|
|
const hasError = validateConfiguration(config);
|
|
|
|
|
|
|
|
if (hasError) {
|
|
|
|
throw new Error('Invalid configuration found in the storage.');
|
|
|
|
}
|
|
|
|
|
|
|
|
return config;
|
|
|
|
} catch (err) {
|
2022-07-12 16:34:54 +00:00
|
|
|
LocalStorage.removeItem(key);
|
2022-06-21 23:20:53 +00:00
|
|
|
console.warn(
|
|
|
|
'Malformed data found for network configuration. Removed and continuing...'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const useConfig = (
|
|
|
|
environment: EnvironmentWithOptionalUrl,
|
|
|
|
updateEnvironment: Dispatch<SetStateAction<Environment>>
|
|
|
|
) => {
|
|
|
|
const [config, setConfig] = useState<Configuration | undefined>(
|
2022-07-12 16:34:54 +00:00
|
|
|
getCachedConfig(environment.VEGA_ENV)
|
2022-06-21 23:20:53 +00:00
|
|
|
);
|
|
|
|
const [status, setStatus] = useState<ConfigStatus>(
|
2022-07-12 16:34:54 +00:00
|
|
|
environment.VEGA_CONFIG_URL ? 'idle' : 'success'
|
2022-06-21 23:20:53 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!config && status === 'idle') {
|
|
|
|
(async () => {
|
|
|
|
setStatus('loading-config');
|
|
|
|
try {
|
|
|
|
const response = await fetch(environment.VEGA_CONFIG_URL ?? '');
|
|
|
|
const configData: Configuration = await response.json();
|
|
|
|
|
|
|
|
if (validateConfiguration(configData)) {
|
|
|
|
setStatus('error-validating-config');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
setConfig({ hosts: configData.hosts });
|
|
|
|
LocalStorage.setItem(
|
2022-07-12 16:34:54 +00:00
|
|
|
getCacheKey(environment.VEGA_ENV),
|
2022-06-21 23:20:53 +00:00
|
|
|
JSON.stringify({ hosts: configData.hosts })
|
|
|
|
);
|
|
|
|
} catch (err) {
|
|
|
|
setStatus('error-loading-config');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
// load config only once per runtime
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
|
|
}, [environment.VEGA_CONFIG_URL, !!config, status, setStatus, setConfig]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (
|
|
|
|
config &&
|
|
|
|
!['loading-node', 'success', 'error-loading-node'].includes(status)
|
|
|
|
) {
|
|
|
|
(async () => {
|
|
|
|
setStatus('loading-node');
|
|
|
|
|
|
|
|
// if there's only one configured node to choose from, set is as the env url
|
|
|
|
if (config.hosts.length === 1) {
|
|
|
|
setStatus('success');
|
|
|
|
updateEnvironment((prevEnvironment) => ({
|
|
|
|
...prevEnvironment,
|
2022-07-12 16:34:54 +00:00
|
|
|
VEGA_URL: prevEnvironment.VEGA_URL || config.hosts[0],
|
2022-06-21 23:20:53 +00:00
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// when there are multiple possible hosts, set the env url to the node which responds first
|
|
|
|
try {
|
|
|
|
const requests = config.hosts.map(requestToNode);
|
|
|
|
const index = await promiseRaceToSuccess(requests);
|
|
|
|
setStatus('success');
|
|
|
|
updateEnvironment((prevEnvironment) => ({
|
|
|
|
...prevEnvironment,
|
2022-07-12 16:34:54 +00:00
|
|
|
VEGA_URL: prevEnvironment.VEGA_URL || config.hosts[index],
|
2022-06-21 23:20:53 +00:00
|
|
|
}));
|
|
|
|
} catch (err) {
|
|
|
|
setStatus('error-loading-node');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
// load config only once per runtime
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
|
|
}, [status, !!config, setStatus, updateEnvironment]);
|
|
|
|
|
|
|
|
return {
|
|
|
|
status,
|
2022-07-12 16:34:54 +00:00
|
|
|
config,
|
2022-06-21 23:20:53 +00:00
|
|
|
};
|
|
|
|
};
|