mirror of
				https://github.com/pelican-dev/panel.git
				synced 2025-10-26 22:46:52 +01:00 
			
		
		
		
	 3fceb588fb
			
		
	
	
		3fceb588fb
		
			
		
	
	
	
	
		
			
			Middleware was removed from the `/` route to redirect users without authentication, so now we need to handle this on the front-end properly.
		
			
				
	
	
		
			201 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			201 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import TransferListener from '@/components/server/TransferListener';
 | |
| import React, { useEffect, useState } from 'react';
 | |
| import { NavLink, Route, Switch, useRouteMatch } from 'react-router-dom';
 | |
| import NavigationBar from '@/components/NavigationBar';
 | |
| import ServerConsole from '@/components/server/ServerConsole';
 | |
| import TransitionRouter from '@/TransitionRouter';
 | |
| import WebsocketHandler from '@/components/server/WebsocketHandler';
 | |
| import { ServerContext } from '@/state/server';
 | |
| import DatabasesContainer from '@/components/server/databases/DatabasesContainer';
 | |
| import FileManagerContainer from '@/components/server/files/FileManagerContainer';
 | |
| import { CSSTransition } from 'react-transition-group';
 | |
| import FileEditContainer from '@/components/server/files/FileEditContainer';
 | |
| import SettingsContainer from '@/components/server/settings/SettingsContainer';
 | |
| import ScheduleContainer from '@/components/server/schedules/ScheduleContainer';
 | |
| import ScheduleEditContainer from '@/components/server/schedules/ScheduleEditContainer';
 | |
| import UsersContainer from '@/components/server/users/UsersContainer';
 | |
| import Can from '@/components/elements/Can';
 | |
| import BackupContainer from '@/components/server/backups/BackupContainer';
 | |
| import Spinner from '@/components/elements/Spinner';
 | |
| import ScreenBlock, { NotFound, ServerError } from '@/components/elements/ScreenBlock';
 | |
| import { httpErrorToHuman } from '@/api/http';
 | |
| import { useStoreState } from 'easy-peasy';
 | |
| import SubNavigation from '@/components/elements/SubNavigation';
 | |
| import NetworkContainer from '@/components/server/network/NetworkContainer';
 | |
| import InstallListener from '@/components/server/InstallListener';
 | |
| import StartupContainer from '@/components/server/startup/StartupContainer';
 | |
| import ErrorBoundary from '@/components/elements/ErrorBoundary';
 | |
| import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
 | |
| import { faExternalLinkAlt } from '@fortawesome/free-solid-svg-icons';
 | |
| import RequireServerPermission from '@/hoc/RequireServerPermission';
 | |
| import ServerInstallSvg from '@/assets/images/server_installing.svg';
 | |
| import ServerRestoreSvg from '@/assets/images/server_restore.svg';
 | |
| import ServerErrorSvg from '@/assets/images/server_error.svg';
 | |
| import { useLocation } from 'react-router';
 | |
| 
 | |
| const ConflictStateRenderer = () => {
 | |
|     const status = ServerContext.useStoreState(state => state.server.data?.status || null);
 | |
|     const isTransferring = ServerContext.useStoreState(state => state.server.data?.isTransferring || false);
 | |
| 
 | |
|     return (
 | |
|         status === 'installing' || status === 'install_failed' ?
 | |
|             <ScreenBlock
 | |
|                 title={'Running Installer'}
 | |
|                 image={ServerInstallSvg}
 | |
|                 message={'Your server should be ready soon, please try again in a few minutes.'}
 | |
|             />
 | |
|             :
 | |
|             status === 'suspended' ?
 | |
|                 <ScreenBlock
 | |
|                     title={'Server Suspended'}
 | |
|                     image={ServerErrorSvg}
 | |
|                     message={'This server is suspended and cannot be accessed.'}
 | |
|                 />
 | |
|                 :
 | |
|                 <ScreenBlock
 | |
|                     title={isTransferring ? 'Transferring' : 'Restoring from Backup'}
 | |
|                     image={ServerRestoreSvg}
 | |
|                     message={isTransferring ? 'Your server is being transfered to a new node, please check back later.' : 'Your server is currently being restored from a backup, please check back in a few minutes.'}
 | |
|                 />
 | |
|     );
 | |
| };
 | |
| 
 | |
| export default () => {
 | |
|     const match = useRouteMatch<{ id: string }>();
 | |
|     const location = useLocation();
 | |
| 
 | |
|     const rootAdmin = useStoreState(state => state.user.data!.rootAdmin);
 | |
|     const [ error, setError ] = useState('');
 | |
| 
 | |
|     const id = ServerContext.useStoreState(state => state.server.data?.id);
 | |
|     const uuid = ServerContext.useStoreState(state => state.server.data?.uuid);
 | |
|     const inConflictState = ServerContext.useStoreState(state => state.server.inConflictState);
 | |
|     const serverId = ServerContext.useStoreState(state => state.server.data?.internalId);
 | |
|     const getServer = ServerContext.useStoreActions(actions => actions.server.getServer);
 | |
|     const clearServerState = ServerContext.useStoreActions(actions => actions.clearServerState);
 | |
| 
 | |
|     useEffect(() => () => {
 | |
|         clearServerState();
 | |
|     }, []);
 | |
| 
 | |
|     useEffect(() => {
 | |
|         setError('');
 | |
| 
 | |
|         getServer(match.params.id)
 | |
|             .catch(error => {
 | |
|                 console.error(error);
 | |
|                 setError(httpErrorToHuman(error));
 | |
|             });
 | |
| 
 | |
|         return () => {
 | |
|             clearServerState();
 | |
|         };
 | |
|     }, [ match.params.id ]);
 | |
| 
 | |
|     return (
 | |
|         <React.Fragment key={'server-router'}>
 | |
|             <NavigationBar/>
 | |
|             {(!uuid || !id) ?
 | |
|                 error ?
 | |
|                     <ServerError message={error}/>
 | |
|                     :
 | |
|                     <Spinner size={'large'} centered/>
 | |
|                 :
 | |
|                 <>
 | |
|                     <CSSTransition timeout={150} classNames={'fade'} appear in>
 | |
|                         <SubNavigation>
 | |
|                             <div>
 | |
|                                 <NavLink to={`${match.url}`} exact>Console</NavLink>
 | |
|                                 <Can action={'file.*'}>
 | |
|                                     <NavLink to={`${match.url}/files`}>File Manager</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'database.*'}>
 | |
|                                     <NavLink to={`${match.url}/databases`}>Databases</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'schedule.*'}>
 | |
|                                     <NavLink to={`${match.url}/schedules`}>Schedules</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'user.*'}>
 | |
|                                     <NavLink to={`${match.url}/users`}>Users</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'backup.*'}>
 | |
|                                     <NavLink to={`${match.url}/backups`}>Backups</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'allocation.*'}>
 | |
|                                     <NavLink to={`${match.url}/network`}>Network</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={'startup.*'}>
 | |
|                                     <NavLink to={`${match.url}/startup`}>Startup</NavLink>
 | |
|                                 </Can>
 | |
|                                 <Can action={[ 'settings.*', 'file.sftp' ]} matchAny>
 | |
|                                     <NavLink to={`${match.url}/settings`}>Settings</NavLink>
 | |
|                                 </Can>
 | |
|                                 {rootAdmin &&
 | |
|                                 <a href={'/admin/servers/view/' + serverId} rel="noreferrer" target={'_blank'}>
 | |
|                                     <FontAwesomeIcon icon={faExternalLinkAlt}/>
 | |
|                                 </a>
 | |
|                                 }
 | |
|                             </div>
 | |
|                         </SubNavigation>
 | |
|                     </CSSTransition>
 | |
|                     <InstallListener/>
 | |
|                     <TransferListener/>
 | |
|                     <WebsocketHandler/>
 | |
|                     {(inConflictState && (!rootAdmin || (rootAdmin && !location.pathname.endsWith(`/server/${id}`)))) ?
 | |
|                         <ConflictStateRenderer/>
 | |
|                         :
 | |
|                         <ErrorBoundary>
 | |
|                             <TransitionRouter>
 | |
|                                 <Switch location={location}>
 | |
|                                     <Route path={`${match.path}`} component={ServerConsole} exact/>
 | |
|                                     <Route path={`${match.path}/files`} exact>
 | |
|                                         <RequireServerPermission permissions={'file.*'}>
 | |
|                                             <FileManagerContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/files/:action(edit|new)`} exact>
 | |
|                                         <Spinner.Suspense>
 | |
|                                             <FileEditContainer/>
 | |
|                                         </Spinner.Suspense>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/databases`} exact>
 | |
|                                         <RequireServerPermission permissions={'database.*'}>
 | |
|                                             <DatabasesContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/schedules`} exact>
 | |
|                                         <RequireServerPermission permissions={'schedule.*'}>
 | |
|                                             <ScheduleContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/schedules/:id`} exact>
 | |
|                                         <ScheduleEditContainer/>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/users`} exact>
 | |
|                                         <RequireServerPermission permissions={'user.*'}>
 | |
|                                             <UsersContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/backups`} exact>
 | |
|                                         <RequireServerPermission permissions={'backup.*'}>
 | |
|                                             <BackupContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/network`} exact>
 | |
|                                         <RequireServerPermission permissions={'allocation.*'}>
 | |
|                                             <NetworkContainer/>
 | |
|                                         </RequireServerPermission>
 | |
|                                     </Route>
 | |
|                                     <Route path={`${match.path}/startup`} component={StartupContainer} exact/>
 | |
|                                     <Route path={`${match.path}/settings`} component={SettingsContainer} exact/>
 | |
|                                     <Route path={'*'} component={NotFound}/>
 | |
|                                 </Switch>
 | |
|                             </TransitionRouter>
 | |
|                         </ErrorBoundary>
 | |
|                     }
 | |
|                 </>
 | |
|             }
 | |
|         </React.Fragment>
 | |
|     );
 | |
| };
 |