import React, { useEffect, useState } from 'react'; import { ServerContext } from '@/state/server'; import Modal from '@/components/elements/Modal'; import tw from 'twin.macro'; import Button from '@/components/elements/Button'; import setSelectedDockerImage from '@/api/server/setSelectedDockerImage'; import FlashMessageRender from '@/components/FlashMessageRender'; import useFlash from '@/plugins/useFlash'; import { SocketEvent, SocketRequest } from '@/components/server/events'; import Select from '@/components/elements/Select'; import useWebsocketEvent from '@/plugins/useWebsocketEvent'; import Can from '@/components/elements/Can'; import getServerStartup from '@/api/swr/getServerStartup'; import InputSpinner from '@/components/elements/InputSpinner'; const MATCH_ERRORS = [ 'minecraft 1.17 requires running the server with java 16 or above', 'minecraft 1.18 requires running the server with java 17 or above', 'java.lang.unsupportedclassversionerror', 'unsupported major.minor version', 'has been compiled by a more recent version of the java runtime', ]; const JavaVersionModalFeature = () => { const [ visible, setVisible ] = useState(false); const [ loading, setLoading ] = useState(false); const [ selectedVersion, setSelectedVersion ] = useState(''); const uuid = ServerContext.useStoreState(state => state.server.data!.uuid); const status = ServerContext.useStoreState(state => state.status.value); const { clearFlashes, clearAndAddHttpError } = useFlash(); const { instance } = ServerContext.useStoreState(state => state.socket); const { data, isValidating, mutate } = getServerStartup(uuid, null, { revalidateOnMount: false }); useEffect(() => { if (!visible) return; mutate().then((value) => { setSelectedVersion(Object.keys(value?.dockerImages || [])[0] || ''); }); }, [ visible ]); useWebsocketEvent(SocketEvent.CONSOLE_OUTPUT, (data) => { if (status === 'running') return; if (MATCH_ERRORS.some(p => data.toLowerCase().includes(p.toLowerCase()))) { setVisible(true); } }); const updateJava = () => { setLoading(true); clearFlashes('feature:javaVersion'); setSelectedDockerImage(uuid, selectedVersion) .then(() => { if (status === 'offline' && instance) { instance.send(SocketRequest.SET_STATE, 'restart'); } setVisible(false); }) .catch(error => clearAndAddHttpError({ key: 'feature:javaVersion', error })) .then(() => setLoading(false)); }; useEffect(() => { clearFlashes('feature:javaVersion'); }, []); return ( <Modal visible={visible} onDismissed={() => setVisible(false)} closeOnBackground={false} showSpinnerOverlay={loading} > <FlashMessageRender key={'feature:javaVersion'} css={tw`mb-4`}/> <h2 css={tw`text-2xl mb-4 text-neutral-100`}>Unsupported Java Version</h2> <p css={tw`mt-4`}> This server is currently running an unsupported version of Java and cannot be started. <Can action={'startup.docker-image'}> Please select a supported version from the list below to continue starting the server. </Can> </p> <Can action={'startup.docker-image'}> <div css={tw`mt-4`}> <InputSpinner visible={!data || isValidating}> <Select disabled={!data} onChange={e => setSelectedVersion(e.target.value)}> {!data ? <option disabled/> : Object.keys((data.dockerImages)).map((key) => ( <option key={key} value={data.dockerImages[key]}>{key}</option> )) } </Select> </InputSpinner> </div> </Can> <div css={tw`mt-8 flex flex-col sm:flex-row justify-end sm:space-x-4 space-y-4 sm:space-y-0`}> <Button isSecondary onClick={() => setVisible(false)} css={tw`w-full sm:w-auto`}> Cancel </Button> <Can action={'startup.docker-image'}> <Button onClick={updateJava} css={tw`w-full sm:w-auto`}> Update Docker Image </Button> </Can> </div> </Modal> ); }; export default JavaVersionModalFeature;