mirror of https://github.com/portainer/portainer
112 lines
3.0 KiB
TypeScript
112 lines
3.0 KiB
TypeScript
|
import { createColumnHelper } from '@tanstack/react-table';
|
||
|
import { Lock, Plus, Trash2 } from 'lucide-react';
|
||
|
|
||
|
import { SecretViewModel } from '@/docker/models/secret';
|
||
|
import { isoDate } from '@/portainer/filters/filters';
|
||
|
|
||
|
import { buildNameColumn } from '@@/datatables/buildNameColumn';
|
||
|
import { Datatable, TableSettingsMenu } from '@@/datatables';
|
||
|
import {
|
||
|
BasicTableSettings,
|
||
|
RefreshableTableSettings,
|
||
|
createPersistedStore,
|
||
|
refreshableSettings,
|
||
|
} from '@@/datatables/types';
|
||
|
import { useTableState } from '@@/datatables/useTableState';
|
||
|
import { TableSettingsMenuAutoRefresh } from '@@/datatables/TableSettingsMenuAutoRefresh';
|
||
|
import { Button } from '@@/buttons';
|
||
|
import { Link } from '@@/Link';
|
||
|
import { useRepeater } from '@@/datatables/useRepeater';
|
||
|
|
||
|
import { createOwnershipColumn } from '../../components/datatable-helpers/createOwnershipColumn';
|
||
|
|
||
|
const columnHelper = createColumnHelper<SecretViewModel>();
|
||
|
|
||
|
const columns = [
|
||
|
buildNameColumn<SecretViewModel>('Name', 'Id', '.secret'),
|
||
|
columnHelper.accessor((item) => isoDate(item.CreatedAt), {
|
||
|
header: 'Creation Date',
|
||
|
}),
|
||
|
createOwnershipColumn<SecretViewModel>(),
|
||
|
];
|
||
|
|
||
|
interface TableSettings extends BasicTableSettings, RefreshableTableSettings {}
|
||
|
|
||
|
const storageKey = 'docker-secrets';
|
||
|
const store = createPersistedStore<TableSettings>(
|
||
|
storageKey,
|
||
|
undefined,
|
||
|
(set) => ({
|
||
|
...refreshableSettings(set),
|
||
|
})
|
||
|
);
|
||
|
|
||
|
export function SecretsDatatable({
|
||
|
dataset,
|
||
|
onRemove,
|
||
|
onRefresh,
|
||
|
}: {
|
||
|
dataset?: Array<SecretViewModel>;
|
||
|
onRemove(items: Array<SecretViewModel>): void;
|
||
|
onRefresh(): Promise<void>;
|
||
|
}) {
|
||
|
const tableState = useTableState(store, storageKey);
|
||
|
useRepeater(tableState.autoRefreshRate, onRefresh);
|
||
|
|
||
|
return (
|
||
|
<Datatable
|
||
|
title="Secrets"
|
||
|
titleIcon={Lock}
|
||
|
columns={columns}
|
||
|
dataset={dataset || []}
|
||
|
isLoading={!dataset}
|
||
|
settingsManager={tableState}
|
||
|
emptyContentLabel="No secret available."
|
||
|
renderTableActions={(selectedItems) => (
|
||
|
<TableActions selectedItems={selectedItems} onRemove={onRemove} />
|
||
|
)}
|
||
|
renderTableSettings={() => (
|
||
|
<TableSettingsMenu>
|
||
|
<TableSettingsMenuAutoRefresh
|
||
|
value={tableState.autoRefreshRate}
|
||
|
onChange={(value) => tableState.setAutoRefreshRate(value)}
|
||
|
/>
|
||
|
</TableSettingsMenu>
|
||
|
)}
|
||
|
/>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
function TableActions({
|
||
|
selectedItems,
|
||
|
onRemove,
|
||
|
}: {
|
||
|
selectedItems: Array<SecretViewModel>;
|
||
|
onRemove(items: Array<SecretViewModel>): void;
|
||
|
}) {
|
||
|
return (
|
||
|
<div className="flex items-center gap-2">
|
||
|
<Button
|
||
|
color="dangerlight"
|
||
|
disabled={selectedItems.length === 0}
|
||
|
onClick={() => onRemove(selectedItems)}
|
||
|
icon={Trash2}
|
||
|
className="!m-0"
|
||
|
data-cy="secret-removeSecretButton"
|
||
|
>
|
||
|
Remove
|
||
|
</Button>
|
||
|
|
||
|
<Button
|
||
|
as={Link}
|
||
|
props={{ to: '.new' }}
|
||
|
icon={Plus}
|
||
|
className="!m-0"
|
||
|
data-cy="secret-addSecretButton"
|
||
|
>
|
||
|
Add secret
|
||
|
</Button>
|
||
|
</div>
|
||
|
);
|
||
|
}
|