mirror of
https://github.com/shankar0123/certctl.git
synced 2026-06-10 21:08:55 +00:00
ff10c85c68
Bug fix: - markNotificationRead was using raw fetch() without auth headers, bypassing the shared client's Authorization header. Moved to api/client.ts to use fetchJSON with proper auth. New action buttons: - CertificatesPage: "New Certificate" modal with form fields - CertificateDetailPage: "Deploy" button with target selector modal, "Archive" button with confirmation - IssuersPage: "Test Connection" and "Delete" per-row actions - TargetsPage: "Delete" per-row action - PoliciesPage: "Enable/Disable" toggle and "Delete" per-row actions New API client functions: - updateCertificate, archiveCertificate, registerAgent, createPolicy, updatePolicy, deletePolicy, getPolicyViolations, createIssuer, testIssuerConnection, deleteIssuer, createTarget, deleteTarget, markNotificationRead Frontend tests (53 tests, 2 files): - client.test.ts: 35 tests covering all API endpoints, auth headers, 401 handling, error parsing, HTTP methods, request bodies - utils.test.ts: 18 tests covering formatDate, formatDateTime, timeAgo, daysUntil, expiryColor CI: Added "Run Frontend Tests" step to frontend-build job Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
97 lines
2.8 KiB
TypeScript
97 lines
2.8 KiB
TypeScript
import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
|
|
import { getTargets, deleteTarget } from '../api/client';
|
|
import PageHeader from '../components/PageHeader';
|
|
import DataTable from '../components/DataTable';
|
|
import type { Column } from '../components/DataTable';
|
|
import StatusBadge from '../components/StatusBadge';
|
|
import ErrorState from '../components/ErrorState';
|
|
import { formatDateTime } from '../api/utils';
|
|
import type { Target } from '../api/types';
|
|
|
|
const typeLabels: Record<string, string> = {
|
|
nginx: 'NGINX',
|
|
f5_bigip: 'F5 BIG-IP',
|
|
iis: 'IIS',
|
|
apache: 'Apache',
|
|
haproxy: 'HAProxy',
|
|
};
|
|
|
|
export default function TargetsPage() {
|
|
const queryClient = useQueryClient();
|
|
|
|
const { data, isLoading, error, refetch } = useQuery({
|
|
queryKey: ['targets'],
|
|
queryFn: () => getTargets(),
|
|
});
|
|
|
|
const deleteMutation = useMutation({
|
|
mutationFn: deleteTarget,
|
|
onSuccess: () => queryClient.invalidateQueries({ queryKey: ['targets'] }),
|
|
});
|
|
|
|
const columns: Column<Target>[] = [
|
|
{
|
|
key: 'name',
|
|
label: 'Target',
|
|
render: (t) => (
|
|
<div>
|
|
<div className="font-medium text-slate-200">{t.name}</div>
|
|
<div className="text-xs text-slate-500 font-mono">{t.id}</div>
|
|
</div>
|
|
),
|
|
},
|
|
{
|
|
key: 'type',
|
|
label: 'Type',
|
|
render: (t) => (
|
|
<span className="badge badge-neutral">{typeLabels[t.type] || t.type}</span>
|
|
),
|
|
},
|
|
{
|
|
key: 'hostname',
|
|
label: 'Hostname',
|
|
render: (t) => <span className="text-slate-300 font-mono text-xs">{t.hostname || '\u2014'}</span>,
|
|
},
|
|
{
|
|
key: 'agent',
|
|
label: 'Agent',
|
|
render: (t) => <span className="text-xs text-slate-400 font-mono">{t.agent_id || '\u2014'}</span>,
|
|
},
|
|
{
|
|
key: 'status',
|
|
label: 'Status',
|
|
render: (t) => <StatusBadge status={t.status} />,
|
|
},
|
|
{
|
|
key: 'created',
|
|
label: 'Created',
|
|
render: (t) => <span className="text-xs text-slate-400">{formatDateTime(t.created_at)}</span>,
|
|
},
|
|
{
|
|
key: 'actions',
|
|
label: '',
|
|
render: (t) => (
|
|
<button
|
|
onClick={(e) => { e.stopPropagation(); if (confirm(`Delete target ${t.name}?`)) deleteMutation.mutate(t.id); }}
|
|
className="text-xs text-red-400 hover:text-red-300 transition-colors"
|
|
>
|
|
Delete
|
|
</button>
|
|
),
|
|
},
|
|
];
|
|
|
|
return (
|
|
<>
|
|
<PageHeader title="Deployment Targets" subtitle={data ? `${data.total} targets` : undefined} />
|
|
<div className="flex-1 overflow-y-auto">
|
|
{error ? (
|
|
<ErrorState error={error as Error} onRetry={() => refetch()} />
|
|
) : (
|
|
<DataTable columns={columns} data={data?.data || []} isLoading={isLoading} emptyMessage="No deployment targets" />
|
|
)}
|
|
</div>
|
|
</>
|
|
);
|
|
}
|