Move actions into context menu, add support for deleting a backup
This commit is contained in:
parent
2eb6ab4d63
commit
9ba6aaebde
10 changed files with 344 additions and 66 deletions
|
@ -2,12 +2,16 @@
|
||||||
|
|
||||||
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
namespace Pterodactyl\Http\Controllers\Api\Client\Servers;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Backup;
|
||||||
use Pterodactyl\Models\Server;
|
use Pterodactyl\Models\Server;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
use Pterodactyl\Services\Backups\DeleteBackupService;
|
||||||
use Pterodactyl\Services\Backups\InitiateBackupService;
|
use Pterodactyl\Services\Backups\InitiateBackupService;
|
||||||
use Pterodactyl\Transformers\Api\Client\BackupTransformer;
|
use Pterodactyl\Transformers\Api\Client\BackupTransformer;
|
||||||
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
use Pterodactyl\Http\Controllers\Api\Client\ClientApiController;
|
||||||
use Pterodactyl\Http\Requests\Api\Client\Servers\Backups\GetBackupsRequest;
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Backups\GetBackupsRequest;
|
||||||
use Pterodactyl\Http\Requests\Api\Client\Servers\Backups\StoreBackupRequest;
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Backups\StoreBackupRequest;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\Servers\Backups\DeleteBackupRequest;
|
||||||
|
|
||||||
class BackupController extends ClientApiController
|
class BackupController extends ClientApiController
|
||||||
{
|
{
|
||||||
|
@ -16,16 +20,23 @@ class BackupController extends ClientApiController
|
||||||
*/
|
*/
|
||||||
private $initiateBackupService;
|
private $initiateBackupService;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var \Pterodactyl\Services\Backups\DeleteBackupService
|
||||||
|
*/
|
||||||
|
private $deleteBackupService;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* BackupController constructor.
|
* BackupController constructor.
|
||||||
*
|
*
|
||||||
|
* @param \Pterodactyl\Services\Backups\DeleteBackupService $deleteBackupService
|
||||||
* @param \Pterodactyl\Services\Backups\InitiateBackupService $initiateBackupService
|
* @param \Pterodactyl\Services\Backups\InitiateBackupService $initiateBackupService
|
||||||
*/
|
*/
|
||||||
public function __construct(InitiateBackupService $initiateBackupService)
|
public function __construct(DeleteBackupService $deleteBackupService, InitiateBackupService $initiateBackupService)
|
||||||
{
|
{
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
|
|
||||||
$this->initiateBackupService = $initiateBackupService;
|
$this->initiateBackupService = $initiateBackupService;
|
||||||
|
$this->deleteBackupService = $deleteBackupService;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -50,7 +61,7 @@ class BackupController extends ClientApiController
|
||||||
* @param \Pterodactyl\Models\Server $server
|
* @param \Pterodactyl\Models\Server $server
|
||||||
* @return array
|
* @return array
|
||||||
*
|
*
|
||||||
* @throws \Exception
|
* @throws \Exception|\Throwable
|
||||||
*/
|
*/
|
||||||
public function store(StoreBackupRequest $request, Server $server)
|
public function store(StoreBackupRequest $request, Server $server)
|
||||||
{
|
{
|
||||||
|
@ -63,15 +74,36 @@ class BackupController extends ClientApiController
|
||||||
->toArray();
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function view()
|
/**
|
||||||
|
* Returns information about a single backup.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Backups\GetBackupsRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Backup $backup
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function view(GetBackupsRequest $request, Server $server, Backup $backup)
|
||||||
{
|
{
|
||||||
|
return $this->fractal->item($backup)
|
||||||
|
->transformWith($this->getTransformer(BackupTransformer::class))
|
||||||
|
->toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function update()
|
/**
|
||||||
|
* Deletes a backup from the panel as well as the remote source where it is currently
|
||||||
|
* being stored.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Http\Requests\Api\Client\Servers\Backups\DeleteBackupRequest $request
|
||||||
|
* @param \Pterodactyl\Models\Server $server
|
||||||
|
* @param \Pterodactyl\Models\Backup $backup
|
||||||
|
* @return \Illuminate\Http\JsonResponse
|
||||||
|
*
|
||||||
|
* @throws \Throwable
|
||||||
|
*/
|
||||||
|
public function delete(DeleteBackupRequest $request, Server $server, Backup $backup)
|
||||||
{
|
{
|
||||||
}
|
$this->deleteBackupService->handle($backup);
|
||||||
|
|
||||||
public function delete()
|
return JsonResponse::create([], JsonResponse::HTTP_NO_CONTENT);
|
||||||
{
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Http\Requests\Api\Client\Servers\Backups;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Permission;
|
||||||
|
use Pterodactyl\Http\Requests\Api\Client\ClientApiRequest;
|
||||||
|
|
||||||
|
class DeleteBackupRequest extends ClientApiRequest
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function permission()
|
||||||
|
{
|
||||||
|
return Permission::ACTION_BACKUP_DELETE;
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,22 +39,19 @@ class DaemonBackupRepository extends DaemonRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a stream of a backup's contents from the Wings instance so that we
|
* Deletes a backup from the daemon.
|
||||||
* do not need to send the user directly to the Daemon.
|
|
||||||
*
|
*
|
||||||
* @param string $backup
|
* @param \Pterodactyl\Models\Backup $backup
|
||||||
* @return \Psr\Http\Message\ResponseInterface
|
* @return \Psr\Http\Message\ResponseInterface
|
||||||
*
|
|
||||||
* @throws \Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException
|
* @throws \Pterodactyl\Exceptions\Http\Connection\DaemonConnectionException
|
||||||
*/
|
*/
|
||||||
public function getBackup(string $backup): ResponseInterface
|
public function delete(Backup $backup): ResponseInterface
|
||||||
{
|
{
|
||||||
Assert::isInstanceOf($this->server, Server::class);
|
Assert::isInstanceOf($this->server, Server::class);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
return $this->getHttpClient()->get(
|
return $this->getHttpClient()->delete(
|
||||||
sprintf('/api/servers/%s/backup/%s', $this->server->uuid, $backup),
|
sprintf('/api/servers/%s/backup/%s', $this->server->uuid, $backup->uuid)
|
||||||
['stream' => true]
|
|
||||||
);
|
);
|
||||||
} catch (TransferException $exception) {
|
} catch (TransferException $exception) {
|
||||||
throw new DaemonConnectionException($exception);
|
throw new DaemonConnectionException($exception);
|
||||||
|
|
58
app/Services/Backups/DeleteBackupService.php
Normal file
58
app/Services/Backups/DeleteBackupService.php
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Pterodactyl\Services\Backups;
|
||||||
|
|
||||||
|
use Pterodactyl\Models\Backup;
|
||||||
|
use Illuminate\Database\ConnectionInterface;
|
||||||
|
use Pterodactyl\Repositories\Eloquent\BackupRepository;
|
||||||
|
use Pterodactyl\Repositories\Wings\DaemonBackupRepository;
|
||||||
|
|
||||||
|
class DeleteBackupService
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var \Pterodactyl\Repositories\Eloquent\BackupRepository
|
||||||
|
*/
|
||||||
|
private $repository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var \Pterodactyl\Repositories\Wings\DaemonBackupRepository
|
||||||
|
*/
|
||||||
|
private $daemonBackupRepository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var \Illuminate\Database\ConnectionInterface
|
||||||
|
*/
|
||||||
|
private $connection;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DeleteBackupService constructor.
|
||||||
|
*
|
||||||
|
* @param \Illuminate\Database\ConnectionInterface $connection
|
||||||
|
* @param \Pterodactyl\Repositories\Eloquent\BackupRepository $repository
|
||||||
|
* @param \Pterodactyl\Repositories\Wings\DaemonBackupRepository $daemonBackupRepository
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
ConnectionInterface $connection,
|
||||||
|
BackupRepository $repository,
|
||||||
|
DaemonBackupRepository $daemonBackupRepository
|
||||||
|
) {
|
||||||
|
$this->repository = $repository;
|
||||||
|
$this->daemonBackupRepository = $daemonBackupRepository;
|
||||||
|
$this->connection = $connection;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Deletes a backup from the system.
|
||||||
|
*
|
||||||
|
* @param \Pterodactyl\Models\Backup $backup
|
||||||
|
* @throws \Throwable
|
||||||
|
*/
|
||||||
|
public function handle(Backup $backup)
|
||||||
|
{
|
||||||
|
$this->connection->transaction(function () use ($backup) {
|
||||||
|
$this->daemonBackupRepository->setServer($backup->server)->delete($backup);
|
||||||
|
|
||||||
|
$this->repository->delete($backup->id);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
9
resources/scripts/api/server/backups/deleteBackup.ts
Normal file
9
resources/scripts/api/server/backups/deleteBackup.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import http from '@/api/http';
|
||||||
|
|
||||||
|
export default (uuid: string, backup: string): Promise<void> => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
http.delete(`/api/client/servers/${uuid}/backups/${backup}`)
|
||||||
|
.then(() => resolve())
|
||||||
|
.catch(reject);
|
||||||
|
});
|
||||||
|
};
|
87
resources/scripts/components/elements/DropdownMenu.tsx
Normal file
87
resources/scripts/components/elements/DropdownMenu.tsx
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
import React, { useEffect, useRef, useState } from 'react';
|
||||||
|
import { CSSTransition } from 'react-transition-group';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
children: React.ReactNode;
|
||||||
|
renderToggle: (onClick: (e: React.MouseEvent<any, MouseEvent>) => void) => React.ReactChild;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DropdownButtonRow = styled.button<{ danger?: boolean }>`
|
||||||
|
${tw`p-2 flex items-center rounded w-full text-neutral-500`};
|
||||||
|
transition: 150ms all ease;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
${props => props.danger
|
||||||
|
? tw`text-red-700 bg-red-100`
|
||||||
|
: tw`text-neutral-700 bg-neutral-100`
|
||||||
|
};
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const DropdownMenu = ({ renderToggle, children }: Props) => {
|
||||||
|
const menu = useRef<HTMLDivElement>(null);
|
||||||
|
const [ posX, setPosX ] = useState(0);
|
||||||
|
const [ visible, setVisible ] = useState(false);
|
||||||
|
|
||||||
|
const onClickHandler = (e: React.MouseEvent<any, MouseEvent>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
!visible && setPosX(e.clientX);
|
||||||
|
setVisible(s => !s);
|
||||||
|
};
|
||||||
|
|
||||||
|
const windowListener = (e: MouseEvent) => {
|
||||||
|
if (e.button === 2 || !visible || !menu.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.target === menu.current || menu.current.contains(e.target as Node)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.target !== menu.current && !menu.current.contains(e.target as Node)) {
|
||||||
|
setVisible(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!visible || !menu.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('click', windowListener);
|
||||||
|
menu.current.setAttribute(
|
||||||
|
'style', `left: ${Math.round(posX - menu.current.clientWidth)}px`,
|
||||||
|
);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener('click', windowListener);
|
||||||
|
}
|
||||||
|
}, [ visible ]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
{renderToggle(onClickHandler)}
|
||||||
|
<CSSTransition
|
||||||
|
timeout={250}
|
||||||
|
in={visible}
|
||||||
|
unmountOnExit={true}
|
||||||
|
classNames={'fade'}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
ref={menu}
|
||||||
|
onClick={e => {
|
||||||
|
e.stopPropagation();
|
||||||
|
setVisible(false);
|
||||||
|
}}
|
||||||
|
className={'absolute bg-white p-2 rounded border border-neutral-700 shadow-lg text-neutral-500 min-w-48'}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</CSSTransition>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DropdownMenu;
|
|
@ -0,0 +1,109 @@
|
||||||
|
import React, { useState } from 'react';
|
||||||
|
import { ServerBackup } from '@/api/server/backups/getServerBackups';
|
||||||
|
import { faEllipsisH } from '@fortawesome/free-solid-svg-icons/faEllipsisH';
|
||||||
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
|
import DropdownMenu, { DropdownButtonRow } from '@/components/elements/DropdownMenu';
|
||||||
|
import { faCloudDownloadAlt } from '@fortawesome/free-solid-svg-icons/faCloudDownloadAlt';
|
||||||
|
import { faTrashAlt } from '@fortawesome/free-solid-svg-icons/faTrashAlt';
|
||||||
|
import { faLock } from '@fortawesome/free-solid-svg-icons/faLock';
|
||||||
|
import getBackupDownloadUrl from '@/api/server/backups/getBackupDownloadUrl';
|
||||||
|
import { httpErrorToHuman } from '@/api/http';
|
||||||
|
import useFlash from '@/plugins/useFlash';
|
||||||
|
import ChecksumModal from '@/components/server/backups/ChecksumModal';
|
||||||
|
import SpinnerOverlay from '@/components/elements/SpinnerOverlay';
|
||||||
|
import useServer from '@/plugins/useServer';
|
||||||
|
import deleteBackup from '@/api/server/backups/deleteBackup';
|
||||||
|
import { ServerContext } from '@/state/server';
|
||||||
|
import ConfirmationModal from '@/components/elements/ConfirmationModal';
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
backup: ServerBackup;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ({ backup }: Props) => {
|
||||||
|
const { uuid } = useServer();
|
||||||
|
const [ loading, setLoading ] = useState(false);
|
||||||
|
const [ visible, setVisible ] = useState(false);
|
||||||
|
const [ deleteVisible, setDeleteVisible ] = useState(false);
|
||||||
|
const { addError, clearFlashes } = useFlash();
|
||||||
|
const removeBackup = ServerContext.useStoreActions(actions => actions.backups.removeBackup);
|
||||||
|
|
||||||
|
const doDownload = () => {
|
||||||
|
setLoading(true);
|
||||||
|
clearFlashes('backups');
|
||||||
|
getBackupDownloadUrl(uuid, backup.uuid)
|
||||||
|
.then(url => {
|
||||||
|
// @ts-ignore
|
||||||
|
window.location = url;
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error(error);
|
||||||
|
addError({ key: 'backups', message: httpErrorToHuman(error) });
|
||||||
|
})
|
||||||
|
.then(() => setLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
|
const doDeletion = () => {
|
||||||
|
setLoading(true);
|
||||||
|
clearFlashes('backups');
|
||||||
|
deleteBackup(uuid, backup.uuid)
|
||||||
|
.then(() => removeBackup(backup.uuid))
|
||||||
|
.catch(error => {
|
||||||
|
console.error(error);
|
||||||
|
addError({ key: 'backups', message: httpErrorToHuman(error) });
|
||||||
|
setLoading(false);
|
||||||
|
setDeleteVisible(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{visible &&
|
||||||
|
<ChecksumModal
|
||||||
|
visible={visible}
|
||||||
|
appear={true}
|
||||||
|
onDismissed={() => setVisible(false)}
|
||||||
|
checksum={backup.sha256Hash}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
{deleteVisible &&
|
||||||
|
<ConfirmationModal
|
||||||
|
title={'Delete this backup?'}
|
||||||
|
buttonText={'Yes, delete backup'}
|
||||||
|
onConfirmed={() => doDeletion()}
|
||||||
|
visible={deleteVisible}
|
||||||
|
onDismissed={() => setDeleteVisible(false)}
|
||||||
|
>
|
||||||
|
Are you sure you wish to delete this backup? This is a permanent operation and the backup cannot
|
||||||
|
be recovered once deleted.
|
||||||
|
</ConfirmationModal>
|
||||||
|
}
|
||||||
|
<SpinnerOverlay visible={loading} fixed={true}/>
|
||||||
|
<DropdownMenu
|
||||||
|
renderToggle={onClick => (
|
||||||
|
<button
|
||||||
|
onClick={onClick}
|
||||||
|
className={'text-neutral-200 transition-color duration-150 hover:text-neutral-100 p-2'}
|
||||||
|
>
|
||||||
|
<FontAwesomeIcon icon={faEllipsisH}/>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className={'text-sm'}>
|
||||||
|
<DropdownButtonRow onClick={() => doDownload()}>
|
||||||
|
<FontAwesomeIcon fixedWidth={true} icon={faCloudDownloadAlt} className={'text-xs'}/>
|
||||||
|
<span className={'ml-2'}>Download</span>
|
||||||
|
</DropdownButtonRow>
|
||||||
|
<DropdownButtonRow onClick={() => setVisible(true)}>
|
||||||
|
<FontAwesomeIcon fixedWidth={true} icon={faLock} className={'text-xs'}/>
|
||||||
|
<span className={'ml-2'}>Checksum</span>
|
||||||
|
</DropdownButtonRow>
|
||||||
|
<DropdownButtonRow danger={true} onClick={() => setDeleteVisible(true)}>
|
||||||
|
<FontAwesomeIcon fixedWidth={true} icon={faTrashAlt} className={'text-xs'}/>
|
||||||
|
<span className={'ml-2'}>Delete</span>
|
||||||
|
</DropdownButtonRow>
|
||||||
|
</div>
|
||||||
|
</DropdownMenu>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
|
@ -5,7 +5,6 @@ import { faArchive } from '@fortawesome/free-solid-svg-icons/faArchive';
|
||||||
import format from 'date-fns/format';
|
import format from 'date-fns/format';
|
||||||
import distanceInWordsToNow from 'date-fns/distance_in_words_to_now';
|
import distanceInWordsToNow from 'date-fns/distance_in_words_to_now';
|
||||||
import Spinner from '@/components/elements/Spinner';
|
import Spinner from '@/components/elements/Spinner';
|
||||||
import { faCloudDownloadAlt } from '@fortawesome/free-solid-svg-icons/faCloudDownloadAlt';
|
|
||||||
import Modal, { RequiredModalProps } from '@/components/elements/Modal';
|
import Modal, { RequiredModalProps } from '@/components/elements/Modal';
|
||||||
import { bytesToHuman } from '@/helpers';
|
import { bytesToHuman } from '@/helpers';
|
||||||
import Can from '@/components/elements/Can';
|
import Can from '@/components/elements/Can';
|
||||||
|
@ -16,30 +15,15 @@ import useFlash from '@/plugins/useFlash';
|
||||||
import { httpErrorToHuman } from '@/api/http';
|
import { httpErrorToHuman } from '@/api/http';
|
||||||
import useWebsocketEvent from '@/plugins/useWebsocketEvent';
|
import useWebsocketEvent from '@/plugins/useWebsocketEvent';
|
||||||
import { ServerContext } from '@/state/server';
|
import { ServerContext } from '@/state/server';
|
||||||
|
import BackupContextMenu from '@/components/server/backups/BackupContextMenu';
|
||||||
|
import { faEllipsisH } from '@fortawesome/free-solid-svg-icons/faEllipsisH';
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
backup: ServerBackup;
|
backup: ServerBackup;
|
||||||
className?: string;
|
className?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const DownloadModal = ({ checksum, ...props }: RequiredModalProps & { checksum: string }) => (
|
|
||||||
<Modal {...props}>
|
|
||||||
<h3 className={'mb-6'}>Verify file checksum</h3>
|
|
||||||
<p className={'text-sm'}>
|
|
||||||
The SHA256 checksum of this file is:
|
|
||||||
</p>
|
|
||||||
<pre className={'mt-2 text-sm p-2 bg-neutral-900 rounded'}>
|
|
||||||
<code className={'block font-mono'}>{checksum}</code>
|
|
||||||
</pre>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
|
|
||||||
export default ({ backup, className }: Props) => {
|
export default ({ backup, className }: Props) => {
|
||||||
const { uuid } = useServer();
|
|
||||||
const { addError, clearFlashes } = useFlash();
|
|
||||||
const [ loading, setLoading ] = useState(false);
|
|
||||||
const [ visible, setVisible ] = useState(false);
|
|
||||||
|
|
||||||
const appendBackup = ServerContext.useStoreActions(actions => actions.backups.appendBackup);
|
const appendBackup = ServerContext.useStoreActions(actions => actions.backups.appendBackup);
|
||||||
|
|
||||||
useWebsocketEvent(`backup completed:${backup.uuid}`, data => {
|
useWebsocketEvent(`backup completed:${backup.uuid}`, data => {
|
||||||
|
@ -56,33 +40,8 @@ export default ({ backup, className }: Props) => {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const getBackupLink = () => {
|
|
||||||
setLoading(true);
|
|
||||||
clearFlashes('backups');
|
|
||||||
getBackupDownloadUrl(uuid, backup.uuid)
|
|
||||||
.then(url => {
|
|
||||||
// @ts-ignore
|
|
||||||
window.location = url;
|
|
||||||
setVisible(true);
|
|
||||||
})
|
|
||||||
.catch(error => {
|
|
||||||
console.error(error);
|
|
||||||
addError({ key: 'backups', message: httpErrorToHuman(error) });
|
|
||||||
})
|
|
||||||
.then(() => setLoading(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={`grey-row-box flex items-center ${className}`}>
|
<div className={`grey-row-box flex items-center ${className}`}>
|
||||||
<SpinnerOverlay visible={loading} fixed={true}/>
|
|
||||||
{visible &&
|
|
||||||
<DownloadModal
|
|
||||||
visible={visible}
|
|
||||||
appear={true}
|
|
||||||
onDismissed={() => setVisible(false)}
|
|
||||||
checksum={backup.sha256Hash}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
<div className={'mr-4'}>
|
<div className={'mr-4'}>
|
||||||
{backup.completedAt ?
|
{backup.completedAt ?
|
||||||
<FontAwesomeIcon icon={faArchive} className={'text-neutral-300'}/>
|
<FontAwesomeIcon icon={faArchive} className={'text-neutral-300'}/>
|
||||||
|
@ -114,15 +73,10 @@ export default ({ backup, className }: Props) => {
|
||||||
<div className={'ml-6'} style={{ marginRight: '-0.5rem' }}>
|
<div className={'ml-6'} style={{ marginRight: '-0.5rem' }}>
|
||||||
{!backup.completedAt ?
|
{!backup.completedAt ?
|
||||||
<div className={'p-2 invisible'}>
|
<div className={'p-2 invisible'}>
|
||||||
<FontAwesomeIcon icon={faCloudDownloadAlt}/>
|
<FontAwesomeIcon icon={faEllipsisH}/>
|
||||||
</div>
|
</div>
|
||||||
:
|
:
|
||||||
<button
|
<BackupContextMenu backup={backup}/>
|
||||||
onClick={() => getBackupLink()}
|
|
||||||
className={'text-sm text-neutral-300 p-2 transition-colors duration-250 hover:text-cyan-400'}
|
|
||||||
>
|
|
||||||
<FontAwesomeIcon icon={faCloudDownloadAlt}/>
|
|
||||||
</button>
|
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</Can>
|
</Can>
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
import React from 'react';
|
||||||
|
import Modal, { RequiredModalProps } from '@/components/elements/Modal';
|
||||||
|
|
||||||
|
const ChecksumModal = ({ checksum, ...props }: RequiredModalProps & { checksum: string }) => (
|
||||||
|
<Modal {...props}>
|
||||||
|
<h3 className={'mb-6'}>Verify file checksum</h3>
|
||||||
|
<p className={'text-sm'}>
|
||||||
|
The SHA256 checksum of this file is:
|
||||||
|
</p>
|
||||||
|
<pre className={'mt-2 text-sm p-2 bg-neutral-900 rounded'}>
|
||||||
|
<code className={'block font-mono'}>{checksum}</code>
|
||||||
|
</pre>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default ChecksumModal;
|
|
@ -92,7 +92,6 @@ Route::group(['prefix' => '/servers/{server}', 'middleware' => [AuthenticateServ
|
||||||
Route::post('/', 'Servers\BackupController@store');
|
Route::post('/', 'Servers\BackupController@store');
|
||||||
Route::get('/{backup}', 'Servers\BackupController@view');
|
Route::get('/{backup}', 'Servers\BackupController@view');
|
||||||
Route::get('/{backup}/download', 'Servers\DownloadBackupController');
|
Route::get('/{backup}/download', 'Servers\DownloadBackupController');
|
||||||
Route::post('/{backup}', 'Servers\BackupController@update');
|
|
||||||
Route::delete('/{backup}', 'Servers\BackupController@delete');
|
Route::delete('/{backup}', 'Servers\BackupController@delete');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue