2026-03-26 16:15:27 +01:00
|
|
|
import { useMemo } from 'react';
|
2026-03-26 12:51:07 +01:00
|
|
|
import { useNavigate } from 'react-router';
|
2026-03-26 16:15:27 +01:00
|
|
|
import { DataTable, Badge, MonoText } from '@cameleer/design-system';
|
2026-03-26 12:51:07 +01:00
|
|
|
import type { Column } from '@cameleer/design-system';
|
2026-03-26 16:15:27 +01:00
|
|
|
import { useAllApplicationConfigs } from '../../api/queries/commands';
|
2026-03-26 12:51:07 +01:00
|
|
|
import type { ApplicationConfig } from '../../api/queries/commands';
|
|
|
|
|
import styles from './AppConfigPage.module.css';
|
|
|
|
|
|
2026-03-26 12:55:19 +01:00
|
|
|
type ConfigRow = ApplicationConfig & { id: string };
|
|
|
|
|
|
2026-03-26 16:15:27 +01:00
|
|
|
type BadgeColor = 'primary' | 'success' | 'warning' | 'error' | 'running' | 'auto';
|
|
|
|
|
|
2026-03-26 12:51:07 +01:00
|
|
|
function timeAgo(iso?: string): string {
|
|
|
|
|
if (!iso) return '\u2014';
|
|
|
|
|
const diff = Date.now() - new Date(iso).getTime();
|
|
|
|
|
const secs = Math.floor(diff / 1000);
|
|
|
|
|
if (secs < 60) return `${secs}s ago`;
|
|
|
|
|
const mins = Math.floor(secs / 60);
|
|
|
|
|
if (mins < 60) return `${mins}m ago`;
|
|
|
|
|
const hours = Math.floor(mins / 60);
|
|
|
|
|
if (hours < 24) return `${hours}h ago`;
|
|
|
|
|
return `${Math.floor(hours / 24)}d ago`;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-26 13:12:56 +01:00
|
|
|
function logLevelColor(level?: string): BadgeColor {
|
2026-03-26 12:51:07 +01:00
|
|
|
switch (level?.toUpperCase()) {
|
|
|
|
|
case 'ERROR': return 'error';
|
|
|
|
|
case 'WARN': return 'warning';
|
|
|
|
|
case 'DEBUG': return 'running';
|
|
|
|
|
default: return 'success';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-26 13:12:56 +01:00
|
|
|
function engineLevelColor(level?: string): BadgeColor {
|
|
|
|
|
switch (level?.toUpperCase()) {
|
|
|
|
|
case 'NONE': return 'error';
|
|
|
|
|
case 'MINIMAL': return 'warning';
|
|
|
|
|
case 'COMPLETE': return 'running';
|
|
|
|
|
default: return 'success';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function payloadColor(mode?: string): BadgeColor {
|
|
|
|
|
switch (mode?.toUpperCase()) {
|
|
|
|
|
case 'INPUT': case 'OUTPUT': return 'warning';
|
|
|
|
|
case 'BOTH': return 'running';
|
|
|
|
|
default: return 'auto';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-26 12:51:07 +01:00
|
|
|
export default function AppConfigPage() {
|
|
|
|
|
const navigate = useNavigate();
|
|
|
|
|
const { data: configs } = useAllApplicationConfigs();
|
|
|
|
|
|
2026-03-26 12:55:19 +01:00
|
|
|
const columns: Column<ConfigRow>[] = useMemo(() => [
|
2026-03-26 12:51:07 +01:00
|
|
|
{
|
|
|
|
|
key: 'application',
|
|
|
|
|
header: 'Application',
|
|
|
|
|
sortable: true,
|
|
|
|
|
render: (_val, row) => <MonoText size="sm">{row.application}</MonoText>,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'logForwardingLevel',
|
|
|
|
|
header: 'Log Level',
|
2026-03-26 13:12:56 +01:00
|
|
|
render: (_val, row) => {
|
|
|
|
|
const val = row.logForwardingLevel ?? 'INFO';
|
|
|
|
|
return <Badge label={val} color={logLevelColor(val)} variant="filled" />;
|
|
|
|
|
},
|
2026-03-26 12:51:07 +01:00
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'engineLevel',
|
|
|
|
|
header: 'Engine Level',
|
2026-03-26 13:12:56 +01:00
|
|
|
render: (_val, row) => {
|
|
|
|
|
const val = row.engineLevel ?? 'REGULAR';
|
|
|
|
|
return <Badge label={val} color={engineLevelColor(val)} variant="filled" />;
|
|
|
|
|
},
|
2026-03-26 12:51:07 +01:00
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'payloadCaptureMode',
|
|
|
|
|
header: 'Payload Capture',
|
2026-03-26 13:12:56 +01:00
|
|
|
render: (_val, row) => {
|
|
|
|
|
const val = row.payloadCaptureMode ?? 'NONE';
|
|
|
|
|
return <Badge label={val} color={payloadColor(val)} variant="filled" />;
|
|
|
|
|
},
|
2026-03-26 12:51:07 +01:00
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'metricsEnabled',
|
|
|
|
|
header: 'Metrics',
|
|
|
|
|
width: '80px',
|
2026-03-26 16:15:27 +01:00
|
|
|
render: (_val, row) => (
|
|
|
|
|
<Badge label={row.metricsEnabled ? 'On' : 'Off'} color={row.metricsEnabled ? 'success' : 'error'} variant="filled" />
|
|
|
|
|
),
|
2026-03-26 12:51:07 +01:00
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'tracedProcessors',
|
|
|
|
|
header: 'Traced',
|
|
|
|
|
width: '70px',
|
|
|
|
|
render: (_val, row) => {
|
|
|
|
|
const count = row.tracedProcessors ? Object.keys(row.tracedProcessors).length : 0;
|
|
|
|
|
return count > 0
|
|
|
|
|
? <Badge label={`${count}`} color="running" variant="filled" />
|
|
|
|
|
: <MonoText size="xs">0</MonoText>;
|
|
|
|
|
},
|
|
|
|
|
},
|
2026-03-26 21:22:59 +01:00
|
|
|
{
|
|
|
|
|
key: 'taps',
|
|
|
|
|
header: 'Taps',
|
|
|
|
|
width: '70px',
|
|
|
|
|
render: (_val, row) => {
|
|
|
|
|
const total = row.taps?.length ?? 0;
|
|
|
|
|
const enabled = row.taps?.filter(t => t.enabled).length ?? 0;
|
|
|
|
|
if (total === 0) return <MonoText size="xs">0</MonoText>;
|
|
|
|
|
return <Badge label={`${enabled}/${total}`} color="running" variant="filled" />;
|
|
|
|
|
},
|
|
|
|
|
},
|
2026-03-26 12:51:07 +01:00
|
|
|
{
|
|
|
|
|
key: 'version',
|
|
|
|
|
header: 'v',
|
|
|
|
|
width: '40px',
|
|
|
|
|
render: (_val, row) => <MonoText size="xs">{row.version}</MonoText>,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
key: 'updatedAt',
|
|
|
|
|
header: 'Updated',
|
|
|
|
|
render: (_val, row) => <MonoText size="xs">{timeAgo(row.updatedAt)}</MonoText>,
|
|
|
|
|
},
|
2026-03-26 16:15:27 +01:00
|
|
|
], []);
|
|
|
|
|
|
|
|
|
|
function handleRowClick(row: ConfigRow) {
|
|
|
|
|
navigate(`/admin/appconfig/${row.application}`);
|
|
|
|
|
}
|
2026-03-26 12:51:07 +01:00
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div>
|
2026-03-26 12:55:19 +01:00
|
|
|
<DataTable<ConfigRow>
|
2026-03-26 12:51:07 +01:00
|
|
|
columns={columns}
|
2026-03-26 12:55:19 +01:00
|
|
|
data={(configs ?? []).map(c => ({ ...c, id: c.application }))}
|
2026-03-26 16:15:27 +01:00
|
|
|
onRowClick={handleRowClick}
|
2026-03-26 12:51:07 +01:00
|
|
|
pageSize={50}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|