import { useEffect, useState } from 'react'; import tw from 'twin.macro'; import getWebsocketToken from '@/api/server/getWebsocketToken'; import ContentContainer from '@/components/elements/ContentContainer'; import Spinner from '@/components/elements/Spinner'; import FadeTransition from '@/components/elements/transitions/FadeTransition'; import { Websocket } from '@/plugins/Websocket'; import { ServerContext } from '@/state/server'; const reconnectErrors = ['jwt: exp claim is invalid', 'jwt: created too far in past (denylist)']; function WebsocketHandler() { let updatingToken = false; const [error, setError] = useState<'connecting' | string>(''); const { connected, instance } = ServerContext.useStoreState(state => state.socket); const uuid = ServerContext.useStoreState(state => state.server.data?.uuid); const setServerStatus = ServerContext.useStoreActions(actions => actions.status.setServerStatus); const { setInstance, setConnectionState } = ServerContext.useStoreActions(actions => actions.socket); const updateToken = (uuid: string, socket: Websocket) => { if (updatingToken) { return; } updatingToken = true; getWebsocketToken(uuid) .then(data => socket.setToken(data.token, true)) .catch(error => console.error(error)) .then(() => { updatingToken = false; }); }; const connect = (uuid: string) => { const socket = new Websocket(); socket.on('auth success', () => setConnectionState(true)); socket.on('SOCKET_CLOSE', () => setConnectionState(false)); socket.on('SOCKET_ERROR', () => { setError('connecting'); setConnectionState(false); }); socket.on('status', status => setServerStatus(status)); socket.on('daemon error', message => { console.warn('Got error message from daemon socket:', message); }); socket.on('token expiring', () => updateToken(uuid, socket)); socket.on('token expired', () => updateToken(uuid, socket)); socket.on('jwt error', (error: string) => { setConnectionState(false); console.warn('JWT validation error from wings:', error); if (reconnectErrors.find(v => error.toLowerCase().indexOf(v) >= 0)) { updateToken(uuid, socket); } else { setError( 'There was an error validating the credentials provided for the websocket. Please refresh the page.', ); } }); socket.on('transfer status', (status: string) => { if (status === 'starting' || status === 'success') { return; } // This code forces a reconnection to the websocket which will connect us to the target node instead of the source node // in order to be able to receive transfer logs from the target node. socket.close(); setError('connecting'); setConnectionState(false); setInstance(null); connect(uuid); }); getWebsocketToken(uuid) .then(data => { // Connect and then set the authentication token. socket.setToken(data.token).connect(data.socket); // Once that is done, set the instance. setInstance(socket); }) .catch(error => console.error(error)); }; useEffect(() => { connected && setError(''); }, [connected]); useEffect(() => { return () => { instance && instance.close(); }; }, [instance]); useEffect(() => { // If there is already an instance or there is no server, just exit out of this process // since we don't need to make a new connection. if (instance || !uuid) { return; } connect(uuid); }, [uuid]); return error ? (
{error === 'connecting' ? ( <>

We're having some trouble connecting to your server, please wait...

) : (

{error}

)}
) : null; } export default WebsocketHandler;