misc_pterodactyl-panel/resources/scripts/components/server/backups/BackupRow.tsx

98 lines
4.2 KiB
TypeScript
Raw Normal View History

import React from 'react';
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
import { faArchive, faEllipsisH, faLock } from '@fortawesome/free-solid-svg-icons';
import { format, formatDistanceToNow } from 'date-fns';
import Spinner from '@/components/elements/Spinner';
import { bytesToHuman } from '@/helpers';
import Can from '@/components/elements/Can';
import useWebsocketEvent from '@/plugins/useWebsocketEvent';
import BackupContextMenu from '@/components/server/backups/BackupContextMenu';
2020-07-04 23:26:07 +00:00
import tw from 'twin.macro';
import GreyRowBox from '@/components/elements/GreyRowBox';
import getServerBackups from '@/api/swr/getServerBackups';
import { ServerBackup } from '@/api/server/types';
import { SocketEvent } from '@/components/server/events';
interface Props {
backup: ServerBackup;
className?: string;
}
2020-04-07 05:25:54 +00:00
export default ({ backup, className }: Props) => {
const { mutate } = getServerBackups();
2020-04-07 05:25:54 +00:00
useWebsocketEvent(`${SocketEvent.BACKUP_COMPLETED}:${backup.uuid}` as SocketEvent, data => {
try {
const parsed = JSON.parse(data);
mutate(data => ({
...data,
items: data.items.map(b => b.uuid !== backup.uuid ? b : ({
...b,
isSuccessful: parsed.is_successful || true,
checksum: (parsed.checksum_type || '') + ':' + (parsed.checksum || ''),
bytes: parsed.file_size || 0,
completedAt: new Date(),
})),
}), false);
} catch (e) {
console.warn(e);
}
});
return (
2020-12-26 17:50:09 +00:00
<GreyRowBox css={tw`flex-wrap md:flex-nowrap items-center`} className={className}>
2020-10-03 23:25:39 +00:00
<div css={tw`flex items-center truncate w-full md:flex-1`}>
<div css={tw`mr-4`}>
{backup.completedAt !== null ?
backup.isLocked ?
<FontAwesomeIcon icon={faLock} css={tw`text-yellow-500`}/>
:
<FontAwesomeIcon icon={faArchive} css={tw`text-neutral-300`}/>
2020-10-03 23:25:39 +00:00
:
<Spinner size={'small'}/>
}
2020-10-03 23:25:39 +00:00
</div>
<div css={tw`flex flex-col truncate`}>
<div css={tw`flex items-center text-sm mb-1`}>
{backup.completedAt !== null && !backup.isSuccessful &&
2020-10-03 23:25:39 +00:00
<span css={tw`bg-red-500 py-px px-2 rounded-full text-white text-xs uppercase border border-red-600 mr-2`}>
Failed
</span>
}
2020-10-11 19:34:48 +00:00
<p css={tw`break-words truncate`}>
2020-10-03 23:25:39 +00:00
{backup.name}
</p>
{(backup.completedAt !== null && backup.isSuccessful) &&
2020-12-26 17:50:09 +00:00
<span css={tw`ml-3 text-neutral-300 text-xs font-extralight hidden sm:inline`}>{bytesToHuman(backup.bytes)}</span>
2020-10-03 23:25:39 +00:00
}
</div>
2020-10-03 23:25:39 +00:00
<p css={tw`mt-1 md:mt-0 text-xs text-neutral-400 font-mono truncate`}>
{backup.checksum}
2020-10-03 23:25:39 +00:00
</p>
</div>
</div>
2020-10-03 23:25:39 +00:00
<div css={tw`flex-1 md:flex-none md:w-48 mt-4 md:mt-0 md:ml-8 md:text-center`}>
<p
2020-07-04 23:26:07 +00:00
title={format(backup.createdAt, 'ddd, MMMM do, yyyy HH:mm:ss')}
css={tw`text-sm`}
>
{formatDistanceToNow(backup.createdAt, { includeSeconds: true, addSuffix: true })}
</p>
2020-07-04 23:26:07 +00:00
<p css={tw`text-2xs text-neutral-500 uppercase mt-1`}>Created</p>
</div>
<Can action={[ 'backup.download', 'backup.restore', 'backup.delete' ]} matchAny>
2020-10-03 23:25:39 +00:00
<div css={tw`mt-4 md:mt-0 ml-6`} style={{ marginRight: '-0.5rem' }}>
{!backup.completedAt ?
2020-07-04 23:26:07 +00:00
<div css={tw`p-2 invisible`}>
<FontAwesomeIcon icon={faEllipsisH}/>
</div>
:
<BackupContextMenu backup={backup}/>
}
</div>
</Can>
2020-07-04 23:26:07 +00:00
</GreyRowBox>
);
};