62c24e3857
- Register AppBarClusterBadge via registerAppBarAction (was dead code) - Add Rook 1.12+ CSI pod labels to CephPodDetailSection alongside legacy labels - Add sidebar entries for Storage Classes and Volumes pages - Add role="dialog", aria-modal, aria-labelledby, and Escape key to all detail drawers - Replace hardcoded hex colors with CSS custom properties for dark/light theme compat - Remove duplicate parseStorageToBytes from OverviewPage (import from k8s.ts) - Add endpoints field to CephObjectStoreStatus interface (remove unsafe cast) - Use ROOK_CEPH_API_GROUP/VERSION constants in API URL construction - Hoist extractJsonData to module level - Remove dead extractPoolFromVolumeHandle function - Fix redundant storageClasses.length guard in OverviewPage - Fix lint indent warnings - Update CLAUDE.md and CHANGELOG.md Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
206 lines
6.2 KiB
TypeScript
206 lines
6.2 KiB
TypeScript
/**
|
|
* StorageClassesPage — lists Rook-Ceph StorageClasses.
|
|
*/
|
|
|
|
import {
|
|
Loader,
|
|
NameValueTable,
|
|
SectionBox,
|
|
SectionHeader,
|
|
SimpleTable,
|
|
StatusLabel,
|
|
} from '@kinvolk/headlamp-plugin/lib/CommonComponents';
|
|
import React, { useState } from 'react';
|
|
import { formatAge, formatStorageType, RookCephStorageClass, storageClassType } from '../api/k8s';
|
|
import { useRookCephContext } from '../api/RookCephDataContext';
|
|
|
|
function StorageClassDetail({
|
|
sc,
|
|
pvCount,
|
|
onClose,
|
|
}: {
|
|
sc: RookCephStorageClass;
|
|
pvCount: number;
|
|
onClose: () => void;
|
|
}) {
|
|
const type = storageClassType(sc);
|
|
return (
|
|
<div
|
|
role="dialog"
|
|
aria-modal="true"
|
|
aria-labelledby="drawer-title-storageclass"
|
|
onKeyDown={e => {
|
|
if (e.key === 'Escape') onClose();
|
|
}}
|
|
style={{
|
|
position: 'fixed',
|
|
top: 0,
|
|
right: 0,
|
|
bottom: 0,
|
|
width: '480px',
|
|
backgroundColor: 'var(--mui-palette-background-paper, #fff)',
|
|
boxShadow: '-4px 0 16px rgba(0,0,0,0.15)',
|
|
zIndex: 1300,
|
|
overflowY: 'auto',
|
|
padding: '24px',
|
|
}}
|
|
>
|
|
<div
|
|
style={{
|
|
display: 'flex',
|
|
justifyContent: 'space-between',
|
|
alignItems: 'center',
|
|
marginBottom: '16px',
|
|
}}
|
|
>
|
|
<strong id="drawer-title-storageclass">{sc.metadata.name}</strong>
|
|
<button
|
|
onClick={onClose}
|
|
aria-label="Close"
|
|
style={{ background: 'none', border: 'none', cursor: 'pointer', fontSize: '18px' }}
|
|
>
|
|
✕
|
|
</button>
|
|
</div>
|
|
<SectionBox title="StorageClass Details">
|
|
<NameValueTable
|
|
rows={[
|
|
{ name: 'Name', value: sc.metadata.name },
|
|
{ name: 'Provisioner', value: sc.provisioner },
|
|
{ name: 'Type', value: formatStorageType(type) },
|
|
{ name: 'Reclaim Policy', value: sc.reclaimPolicy ?? '—' },
|
|
{ name: 'Volume Binding Mode', value: sc.volumeBindingMode ?? '—' },
|
|
{
|
|
name: 'Volume Expansion',
|
|
value: sc.allowVolumeExpansion ? 'Allowed' : 'Not allowed',
|
|
},
|
|
{ name: 'Age', value: formatAge(sc.metadata.creationTimestamp) },
|
|
{ name: 'Bound PVs', value: String(pvCount) },
|
|
]}
|
|
/>
|
|
</SectionBox>
|
|
{sc.parameters && Object.keys(sc.parameters).length > 0 && (
|
|
<SectionBox title="Parameters">
|
|
<NameValueTable
|
|
rows={Object.entries(sc.parameters).map(([k, v]) => ({ name: k, value: v ?? '—' }))}
|
|
/>
|
|
</SectionBox>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function StorageClassesPage() {
|
|
const { storageClasses, persistentVolumes, loading, error } = useRookCephContext();
|
|
const [selected, setSelected] = useState<RookCephStorageClass | null>(null);
|
|
|
|
if (loading) return <Loader title="Loading Rook-Ceph storage classes..." />;
|
|
|
|
const pvCountByClass = new Map<string, number>();
|
|
for (const pv of persistentVolumes) {
|
|
const sc = pv.spec.storageClassName ?? '';
|
|
pvCountByClass.set(sc, (pvCountByClass.get(sc) ?? 0) + 1);
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<SectionHeader title="Rook-Ceph Storage Classes" />
|
|
|
|
{error && (
|
|
<SectionBox title="Error">
|
|
<NameValueTable
|
|
rows={[{ name: 'Status', value: <StatusLabel status="error">{error}</StatusLabel> }]}
|
|
/>
|
|
</SectionBox>
|
|
)}
|
|
|
|
{storageClasses.length === 0 ? (
|
|
<SectionBox title="No Storage Classes">
|
|
<NameValueTable
|
|
rows={[
|
|
{
|
|
name: 'Status',
|
|
value:
|
|
'No Rook-Ceph StorageClasses found. Ensure CephBlockPool and CephFilesystem resources exist.',
|
|
},
|
|
]}
|
|
/>
|
|
</SectionBox>
|
|
) : (
|
|
<SectionBox title={`Storage Classes (${storageClasses.length})`}>
|
|
<SimpleTable
|
|
columns={[
|
|
{
|
|
label: 'Name',
|
|
getter: (sc: RookCephStorageClass) => (
|
|
<button
|
|
onClick={() => setSelected(sc)}
|
|
style={{
|
|
border: 'none',
|
|
background: 'transparent',
|
|
color: 'var(--link-color, #1976d2)',
|
|
cursor: 'pointer',
|
|
textDecoration: 'underline',
|
|
padding: 0,
|
|
font: 'inherit',
|
|
}}
|
|
>
|
|
{sc.metadata.name}
|
|
</button>
|
|
),
|
|
},
|
|
{
|
|
label: 'Type',
|
|
getter: (sc: RookCephStorageClass) => (
|
|
<StatusLabel status="success">
|
|
{formatStorageType(storageClassType(sc))}
|
|
</StatusLabel>
|
|
),
|
|
},
|
|
{ label: 'Provisioner', getter: (sc: RookCephStorageClass) => sc.provisioner },
|
|
{
|
|
label: 'Pool',
|
|
getter: (sc: RookCephStorageClass) => sc.parameters?.['pool'] ?? '—',
|
|
},
|
|
{ label: 'Reclaim', getter: (sc: RookCephStorageClass) => sc.reclaimPolicy ?? '—' },
|
|
{
|
|
label: 'Expansion',
|
|
getter: (sc: RookCephStorageClass) => (sc.allowVolumeExpansion ? 'Yes' : 'No'),
|
|
},
|
|
{
|
|
label: 'PVs',
|
|
getter: (sc: RookCephStorageClass) =>
|
|
String(pvCountByClass.get(sc.metadata.name) ?? 0),
|
|
},
|
|
{
|
|
label: 'Age',
|
|
getter: (sc: RookCephStorageClass) => formatAge(sc.metadata.creationTimestamp),
|
|
},
|
|
]}
|
|
data={storageClasses}
|
|
/>
|
|
</SectionBox>
|
|
)}
|
|
|
|
{selected && (
|
|
<>
|
|
<div
|
|
style={{
|
|
position: 'fixed',
|
|
inset: 0,
|
|
backgroundColor: 'rgba(0,0,0,0.3)',
|
|
zIndex: 1299,
|
|
}}
|
|
onClick={() => setSelected(null)}
|
|
/>
|
|
<StorageClassDetail
|
|
sc={selected}
|
|
pvCount={pvCountByClass.get(selected.metadata.name) ?? 0}
|
|
onClose={() => setSelected(null)}
|
|
/>
|
|
</>
|
|
)}
|
|
</>
|
|
);
|
|
}
|