@calltelemetry/cli 0.4.6 → 0.4.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (61) hide show
  1. package/dist/commands/loadtest.d.ts +3 -0
  2. package/dist/commands/loadtest.d.ts.map +1 -0
  3. package/dist/commands/loadtest.js +22 -0
  4. package/dist/commands/loadtest.js.map +1 -0
  5. package/dist/index.js +3 -1
  6. package/dist/index.js.map +1 -1
  7. package/dist/lib/curri-xml.d.ts +21 -0
  8. package/dist/lib/curri-xml.d.ts.map +1 -0
  9. package/dist/lib/curri-xml.js +53 -0
  10. package/dist/lib/curri-xml.js.map +1 -0
  11. package/dist/lib/loadtest-stats.d.ts +91 -0
  12. package/dist/lib/loadtest-stats.d.ts.map +1 -0
  13. package/dist/lib/loadtest-stats.js +23 -0
  14. package/dist/lib/loadtest-stats.js.map +1 -0
  15. package/dist/lib/loadtest.d.ts +14 -0
  16. package/dist/lib/loadtest.d.ts.map +1 -0
  17. package/dist/lib/loadtest.js +112 -0
  18. package/dist/lib/loadtest.js.map +1 -0
  19. package/dist/lib/rpc-scripts.d.ts +2 -0
  20. package/dist/lib/rpc-scripts.d.ts.map +1 -1
  21. package/dist/lib/rpc-scripts.js +29 -0
  22. package/dist/lib/rpc-scripts.js.map +1 -1
  23. package/dist/ui/components/Sparkline.d.ts +15 -0
  24. package/dist/ui/components/Sparkline.d.ts.map +1 -0
  25. package/dist/ui/components/Sparkline.js +21 -0
  26. package/dist/ui/components/Sparkline.js.map +1 -0
  27. package/dist/ui/components/index.d.ts +1 -0
  28. package/dist/ui/components/index.d.ts.map +1 -1
  29. package/dist/ui/components/index.js +1 -0
  30. package/dist/ui/components/index.js.map +1 -1
  31. package/dist/ui/hooks/index.d.ts +2 -0
  32. package/dist/ui/hooks/index.d.ts.map +1 -1
  33. package/dist/ui/hooks/index.js +1 -0
  34. package/dist/ui/hooks/index.js.map +1 -1
  35. package/dist/ui/hooks/useLoadTest.d.ts +11 -0
  36. package/dist/ui/hooks/useLoadTest.d.ts.map +1 -0
  37. package/dist/ui/hooks/useLoadTest.js +38 -0
  38. package/dist/ui/hooks/useLoadTest.js.map +1 -0
  39. package/dist/ui/views/LoadTestConfigView.d.ts +31 -0
  40. package/dist/ui/views/LoadTestConfigView.d.ts.map +1 -0
  41. package/dist/ui/views/LoadTestConfigView.js +111 -0
  42. package/dist/ui/views/LoadTestConfigView.js.map +1 -0
  43. package/dist/ui/views/LoadTestRunView.d.ts +10 -0
  44. package/dist/ui/views/LoadTestRunView.d.ts.map +1 -0
  45. package/dist/ui/views/LoadTestRunView.js +71 -0
  46. package/dist/ui/views/LoadTestRunView.js.map +1 -0
  47. package/dist/ui/views/LoadTestSummaryView.d.ts +9 -0
  48. package/dist/ui/views/LoadTestSummaryView.d.ts.map +1 -0
  49. package/dist/ui/views/LoadTestSummaryView.js +72 -0
  50. package/dist/ui/views/LoadTestSummaryView.js.map +1 -0
  51. package/dist/ui/views/LoadTestView.d.ts +8 -0
  52. package/dist/ui/views/LoadTestView.d.ts.map +1 -0
  53. package/dist/ui/views/LoadTestView.js +43 -0
  54. package/dist/ui/views/LoadTestView.js.map +1 -0
  55. package/dist/ui/views/MainMenu.d.ts.map +1 -1
  56. package/dist/ui/views/MainMenu.js +83 -21
  57. package/dist/ui/views/MainMenu.js.map +1 -1
  58. package/dist/ui/views/StatusView.d.ts.map +1 -1
  59. package/dist/ui/views/StatusView.js +134 -21
  60. package/dist/ui/views/StatusView.js.map +1 -1
  61. package/package.json +2 -1
@@ -0,0 +1,38 @@
1
+ import { useState, useCallback, useRef } from 'react';
2
+ export function useLoadTest() {
3
+ const [phase, setPhase] = useState('idle');
4
+ const [snapshot, setSnapshot] = useState(null);
5
+ const [results, setResults] = useState(null);
6
+ const [error, setError] = useState(null);
7
+ const handleRef = useRef(null);
8
+ const start = useCallback(async (config) => {
9
+ setPhase('running');
10
+ setSnapshot(null);
11
+ setResults(null);
12
+ setError(null);
13
+ try {
14
+ // Dynamic import to avoid loading autocannon until needed
15
+ const { startLoadTest } = await import('../../lib/loadtest.js');
16
+ const handle = startLoadTest(config, (snap) => setSnapshot(snap));
17
+ handleRef.current = handle;
18
+ const res = await handle.promise;
19
+ handleRef.current = null;
20
+ setResults(res);
21
+ setPhase('complete');
22
+ }
23
+ catch (err) {
24
+ handleRef.current = null;
25
+ setError(err instanceof Error ? err.message : String(err));
26
+ setPhase('error');
27
+ }
28
+ }, []);
29
+ const stop = useCallback(() => {
30
+ if (handleRef.current) {
31
+ handleRef.current.stop();
32
+ handleRef.current = null;
33
+ setPhase('stopped');
34
+ }
35
+ }, []);
36
+ return { phase, snapshot, results, error, start, stop };
37
+ }
38
+ //# sourceMappingURL=useLoadTest.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useLoadTest.js","sourceRoot":"","sources":["../../../src/ui/hooks/useLoadTest.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AAKtD,MAAM,UAAU,WAAW;IACzB,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAgB,MAAM,CAAC,CAAC;IAC1D,MAAM,CAAC,QAAQ,EAAE,WAAW,CAAC,GAAG,QAAQ,CAA0B,IAAI,CAAC,CAAC;IACxE,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAyB,IAAI,CAAC,CAAC;IACrE,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAgB,IAAI,CAAC,CAAC;IACxD,MAAM,SAAS,GAAG,MAAM,CAAwB,IAAI,CAAC,CAAC;IAEtD,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,EAAE,MAAsB,EAAE,EAAE;QACzD,QAAQ,CAAC,SAAS,CAAC,CAAC;QACpB,WAAW,CAAC,IAAI,CAAC,CAAC;QAClB,UAAU,CAAC,IAAI,CAAC,CAAC;QACjB,QAAQ,CAAC,IAAI,CAAC,CAAC;QAEf,IAAI,CAAC;YACH,0DAA0D;YAC1D,MAAM,EAAE,aAAa,EAAE,GAAG,MAAM,MAAM,CAAC,uBAAuB,CAAC,CAAC;YAChE,MAAM,MAAM,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC;YAClE,SAAS,CAAC,OAAO,GAAG,MAAM,CAAC;YAE3B,MAAM,GAAG,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC;YACjC,SAAS,CAAC,OAAO,GAAG,IAAI,CAAC;YACzB,UAAU,CAAC,GAAG,CAAC,CAAC;YAChB,QAAQ,CAAC,UAAU,CAAC,CAAC;QACvB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,SAAS,CAAC,OAAO,GAAG,IAAI,CAAC;YACzB,QAAQ,CAAC,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC;YAC3D,QAAQ,CAAC,OAAO,CAAC,CAAC;QACpB,CAAC;IACH,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,MAAM,IAAI,GAAG,WAAW,CAAC,GAAG,EAAE;QAC5B,IAAI,SAAS,CAAC,OAAO,EAAE,CAAC;YACtB,SAAS,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;YACzB,SAAS,CAAC,OAAO,GAAG,IAAI,CAAC;YACzB,QAAQ,CAAC,SAAS,CAAC,CAAC;QACtB,CAAC;IACH,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;AAC1D,CAAC"}
@@ -0,0 +1,31 @@
1
+ import type { CurriPolicy } from '../../lib/loadtest-stats.js';
2
+ export interface LoadTestCliOptions {
3
+ policy?: string;
4
+ rate?: string;
5
+ duration?: string;
6
+ connections?: string;
7
+ calling?: string;
8
+ called?: string;
9
+ device?: string;
10
+ trigger?: string;
11
+ url?: string;
12
+ }
13
+ export interface LoadTestUserConfig {
14
+ policy: CurriPolicy;
15
+ url: string;
16
+ rate: number;
17
+ duration: number;
18
+ connections: number;
19
+ callingNumber: string;
20
+ calledNumber: string;
21
+ deviceName: string;
22
+ triggerType: string;
23
+ }
24
+ interface LoadTestConfigViewProps {
25
+ onBack?: () => void;
26
+ onStart: (config: LoadTestUserConfig) => void;
27
+ cliOptions?: LoadTestCliOptions;
28
+ }
29
+ export declare function LoadTestConfigView({ onBack, onStart, cliOptions }: LoadTestConfigViewProps): import("react/jsx-runtime").JSX.Element;
30
+ export {};
31
+ //# sourceMappingURL=LoadTestConfigView.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestConfigView.d.ts","sourceRoot":"","sources":["../../../src/ui/views/LoadTestConfigView.tsx"],"names":[],"mappings":"AAOA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,6BAA6B,CAAC;AAa/D,MAAM,WAAW,kBAAkB;IACjC,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,GAAG,CAAC,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,WAAW,CAAC;IACpB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,UAAU,uBAAuB;IAC/B,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;IACpB,OAAO,EAAE,CAAC,MAAM,EAAE,kBAAkB,KAAK,IAAI,CAAC;IAC9C,UAAU,CAAC,EAAE,kBAAkB,CAAC;CACjC;AAED,wBAAgB,kBAAkB,CAAC,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,EAAE,EAAE,uBAAuB,2CA6J1F"}
@@ -0,0 +1,111 @@
1
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { useState, useCallback, useEffect } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ import InkTextInput from 'ink-text-input';
5
+ import { AppShell, Section, SelectMenu, Spinner, StatusBox } from '../components/index.js';
6
+ import { useAsyncTask } from '../hooks/index.js';
7
+ import { rpc } from '../../lib/rpc.js';
8
+ import { CURRI_POLICIES_RPC } from '../../lib/rpc-scripts.js';
9
+ import { DEFAULT_CURRI_URL } from '../../lib/loadtest.js';
10
+ export function LoadTestConfigView({ onBack, onStart, cliOptions }) {
11
+ const [phase, setPhase] = useState('select_policy');
12
+ const [selectedPolicy, setSelectedPolicy] = useState(null);
13
+ const [focusIndex, setFocusIndex] = useState(0);
14
+ const [fields, setFields] = useState([
15
+ { label: 'Rate (req/s)', key: 'rate', value: cliOptions?.rate ?? '100', placeholder: '100' },
16
+ { label: 'Duration (s)', key: 'duration', value: cliOptions?.duration ?? '30', placeholder: '30' },
17
+ { label: 'Connections', key: 'connections', value: cliOptions?.connections ?? '10', placeholder: '10' },
18
+ { label: 'Calling #', key: 'calling', value: cliOptions?.calling ?? 'random', placeholder: 'random' },
19
+ { label: 'Called #', key: 'called', value: cliOptions?.called ?? 'random', placeholder: 'random' },
20
+ { label: 'Device', key: 'device', value: cliOptions?.device ?? 'random', placeholder: 'random' },
21
+ { label: 'Trigger', key: 'trigger', value: cliOptions?.trigger ?? 'translationpattern', placeholder: 'translationpattern' },
22
+ { label: 'Target URL', key: 'url', value: cliOptions?.url ?? DEFAULT_CURRI_URL, placeholder: DEFAULT_CURRI_URL },
23
+ ]);
24
+ // Fetch policies via RPC
25
+ const policyTask = useAsyncTask(async () => {
26
+ const output = await rpc(CURRI_POLICIES_RPC);
27
+ // Find JSON line in output
28
+ const lines = output.split('\n');
29
+ for (const line of lines) {
30
+ const trimmed = line.trim();
31
+ if (trimmed.startsWith('{')) {
32
+ const parsed = JSON.parse(trimmed);
33
+ if (parsed.error)
34
+ throw new Error(parsed.error);
35
+ return parsed.policies;
36
+ }
37
+ }
38
+ throw new Error('No JSON response from RPC');
39
+ }, true);
40
+ // Auto-select policy from CLI options
41
+ useEffect(() => {
42
+ if (cliOptions?.policy && policyTask.status === 'done' && policyTask.data) {
43
+ const match = policyTask.data.find((p) => p.name.toLowerCase() === cliOptions.policy.toLowerCase());
44
+ if (match) {
45
+ setSelectedPolicy(match);
46
+ setPhase('configure');
47
+ }
48
+ }
49
+ }, [cliOptions?.policy, policyTask.status, policyTask.data]);
50
+ const handlePolicySelect = useCallback((item) => {
51
+ if (policyTask.data) {
52
+ const policy = policyTask.data.find((p) => String(p.id) === item.value);
53
+ if (policy) {
54
+ setSelectedPolicy(policy);
55
+ setPhase('configure');
56
+ }
57
+ }
58
+ }, [policyTask.data]);
59
+ const updateField = useCallback((index, value) => {
60
+ setFields((prev) => {
61
+ const next = [...prev];
62
+ next[index] = { ...next[index], value };
63
+ return next;
64
+ });
65
+ }, []);
66
+ const handleStart = useCallback(() => {
67
+ if (!selectedPolicy)
68
+ return;
69
+ const fieldMap = Object.fromEntries(fields.map((f) => [f.key, f.value]));
70
+ onStart({
71
+ policy: selectedPolicy,
72
+ url: fieldMap['url'] ?? DEFAULT_CURRI_URL,
73
+ rate: parseInt(fieldMap['rate'] ?? '100', 10) || 100,
74
+ duration: parseInt(fieldMap['duration'] ?? '30', 10) || 30,
75
+ connections: parseInt(fieldMap['connections'] ?? '10', 10) || 10,
76
+ callingNumber: fieldMap['calling'] ?? 'random',
77
+ calledNumber: fieldMap['called'] ?? 'random',
78
+ deviceName: fieldMap['device'] ?? 'random',
79
+ triggerType: fieldMap['trigger'] ?? 'translationpattern',
80
+ });
81
+ }, [selectedPolicy, fields, onStart]);
82
+ useInput((input, key) => {
83
+ if (phase === 'configure') {
84
+ if (key.upArrow || (key.shift && key.tab)) {
85
+ setFocusIndex((prev) => (prev > 0 ? prev - 1 : fields.length - 1));
86
+ }
87
+ else if (key.downArrow || key.tab) {
88
+ setFocusIndex((prev) => (prev < fields.length - 1 ? prev + 1 : 0));
89
+ }
90
+ else if (key.return) {
91
+ handleStart();
92
+ }
93
+ else if (key.escape) {
94
+ setPhase('select_policy');
95
+ }
96
+ }
97
+ }, { isActive: phase === 'configure' });
98
+ // Build menu items from policies
99
+ const policyItems = (policyTask.data ?? []).map((p) => ({
100
+ label: p.name,
101
+ description: `(${p.org_name})`,
102
+ value: String(p.id),
103
+ }));
104
+ return (_jsx(AppShell, { command: "ct loadtest", onBack: onBack, children: _jsxs(Section, { title: "CURRI Load Test", children: [phase === 'select_policy' && (_jsxs(_Fragment, { children: [policyTask.status === 'pending' && _jsx(Spinner, { label: "Fetching policies..." }), policyTask.status === 'error' && (_jsx(StatusBox, { variant: "error", title: "Failed to fetch policies", lines: [policyTask.error?.message ?? 'Unknown error'] })), policyTask.status === 'done' && policyItems.length === 0 && (_jsx(StatusBox, { variant: "warning", title: "No policies found", lines: ['No CURRI policies are configured. Create a policy in the web UI first.'] })), policyTask.status === 'done' && policyItems.length > 0 && (_jsxs(_Fragment, { children: [_jsx(Text, { bold: true, children: " Select Policy" }), _jsx(SelectMenu, { items: policyItems, onSelect: handlePolicySelect, onExit: onBack })] }))] })), phase === 'configure' && selectedPolicy && (_jsxs(Box, { flexDirection: "column", paddingLeft: 2, children: [_jsxs(Box, { marginBottom: 1, children: [_jsx(Text, { bold: true, color: "cyan", children: "Policy: " }), _jsx(Text, { children: selectedPolicy.name }), _jsx(Text, { dimColor: true, children: ` (${selectedPolicy.org_name})` })] }), fields.map((field, i) => (_jsxs(Box, { children: [_jsx(Text, { bold: true, children: field.label.padEnd(14) }), i === focusIndex ? (_jsx(InkTextInput, { value: field.value, onChange: (v) => updateField(i, v), onSubmit: () => {
105
+ if (i < fields.length - 1)
106
+ setFocusIndex(i + 1);
107
+ else
108
+ handleStart();
109
+ }, placeholder: field.placeholder })) : (_jsx(Text, { dimColor: field.value === field.placeholder, children: field.value || field.placeholder }))] }, field.key))), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: '↑↓ Navigate Enter Start Esc Back' }) })] }))] }) }));
110
+ }
111
+ //# sourceMappingURL=LoadTestConfigView.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestConfigView.js","sourceRoot":"","sources":["../../../src/ui/views/LoadTestConfigView.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import type { LoadTestResults } from '../../lib/loadtest-stats.js';
2
+ import type { LoadTestUserConfig } from './LoadTestConfigView.js';
3
+ interface LoadTestRunViewProps {
4
+ userConfig: LoadTestUserConfig;
5
+ onComplete: (results: LoadTestResults) => void;
6
+ onBack?: () => void;
7
+ }
8
+ export declare function LoadTestRunView({ userConfig, onComplete, onBack }: LoadTestRunViewProps): import("react/jsx-runtime").JSX.Element;
9
+ export {};
10
+ //# sourceMappingURL=LoadTestRunView.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestRunView.d.ts","sourceRoot":"","sources":["../../../src/ui/views/LoadTestRunView.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAkB,eAAe,EAAoB,MAAM,6BAA6B,CAAC;AACrG,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AAElE,UAAU,oBAAoB;IAC5B,UAAU,EAAE,kBAAkB,CAAC;IAC/B,UAAU,EAAE,CAAC,OAAO,EAAE,eAAe,KAAK,IAAI,CAAC;IAC/C,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;CACrB;AAmBD,wBAAgB,eAAe,CAAC,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,EAAE,oBAAoB,2CA2JvF"}
@@ -0,0 +1,71 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { useEffect } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ import { AppShell, Section, MeterBar, StatusLine, Sparkline } from '../components/index.js';
5
+ import { useLoadTest } from '../hooks/index.js';
6
+ function formatMs(ms) {
7
+ if (ms < 1)
8
+ return `${(ms * 1000).toFixed(0)}μs`;
9
+ if (ms < 1000)
10
+ return `${ms.toFixed(1)}ms`;
11
+ return `${(ms / 1000).toFixed(2)}s`;
12
+ }
13
+ function formatNum(n) {
14
+ if (n >= 1_000_000)
15
+ return `${(n / 1_000_000).toFixed(1)}M`;
16
+ if (n >= 1_000)
17
+ return `${(n / 1_000).toFixed(0).replace(/\B(?=(\d{3})+(?!\d))/g, ',')}`;
18
+ return String(n);
19
+ }
20
+ /** Latency bar: maps a latency value to % of a 20ms ceiling. */
21
+ function latencyPercent(ms, ceiling = 20) {
22
+ return Math.min(100, (ms / ceiling) * 100);
23
+ }
24
+ export function LoadTestRunView({ userConfig, onComplete, onBack }) {
25
+ const { phase, snapshot, results, error, start, stop } = useLoadTest();
26
+ const config = {
27
+ url: userConfig.url,
28
+ apiKey: userConfig.policy.api_key,
29
+ policyName: userConfig.policy.name,
30
+ rate: userConfig.rate,
31
+ duration: userConfig.duration,
32
+ connections: userConfig.connections,
33
+ callingNumber: userConfig.callingNumber,
34
+ calledNumber: userConfig.calledNumber,
35
+ deviceName: userConfig.deviceName,
36
+ triggerType: userConfig.triggerType,
37
+ };
38
+ useEffect(() => {
39
+ if (phase === 'idle') {
40
+ start(config);
41
+ }
42
+ }, []); // eslint-disable-line react-hooks/exhaustive-deps
43
+ useEffect(() => {
44
+ if (phase === 'complete' && results) {
45
+ onComplete(results);
46
+ }
47
+ }, [phase, results, onComplete]);
48
+ // Ctrl+C / q to stop early
49
+ useInput((input, key) => {
50
+ if ((input === 'q' || (key.ctrl && input === 'c')) && phase === 'running') {
51
+ stop();
52
+ }
53
+ if (phase === 'error' && (input === 'q' || key.escape)) {
54
+ onBack?.();
55
+ }
56
+ });
57
+ const elapsed = snapshot?.elapsed ?? 0;
58
+ const progressPct = config.duration > 0 ? Math.min(100, (elapsed / config.duration) * 100) : 0;
59
+ const avgRate = snapshot && elapsed > 0
60
+ ? Math.round(snapshot.totalRequests / elapsed)
61
+ : 0;
62
+ const totalReqs = snapshot?.totalRequests ?? 0;
63
+ const successRate = totalReqs > 0
64
+ ? ((snapshot.successCount / totalReqs) * 100).toFixed(1)
65
+ : '0.0';
66
+ const errorRate = totalReqs > 0
67
+ ? ((snapshot.errorCount / totalReqs) * 100).toFixed(1)
68
+ : '0.0';
69
+ return (_jsx(AppShell, { command: "ct loadtest", onBack: onBack, children: _jsxs(Section, { title: `CURRI Load Test — ${userConfig.policy.name}`, children: [phase === 'error' && (_jsxs(Box, { flexDirection: "column", paddingLeft: 2, children: [_jsx(Text, { color: "red", bold: true, children: "Load test failed" }), _jsx(Text, { color: "red", children: error }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: "q Back" }) })] })), (phase === 'running' || phase === 'stopped') && (_jsxs(Box, { flexDirection: "column", paddingLeft: 2, children: [_jsx(MeterBar, { label: "Progress", percent: progressPct, width: 24, detail: `${elapsed}s / ${config.duration}s`, thresholds: [101, 102] }), _jsx(Sparkline, { data: snapshot?.rateHistory ?? [], width: 24, label: "Rate", detail: `avg ${avgRate} req/s`, color: "cyan" }), _jsx(Box, { marginTop: 1 }), _jsx(Text, { bold: true, children: " Latency" }), _jsxs(Box, { flexDirection: "column", paddingLeft: 2, children: [_jsxs(Box, { children: [_jsx(Text, { children: 'p50'.padEnd(6) }), _jsx(Text, { color: "green", bold: true, children: formatMs(snapshot?.p50 ?? 0).padEnd(10) })] }), _jsxs(Box, { children: [_jsx(Text, { children: 'p95'.padEnd(6) }), _jsx(Text, { color: "yellow", bold: true, children: formatMs(snapshot?.p95 ?? 0).padEnd(10) }), _jsx(MeterBar, { label: "", percent: latencyPercent(snapshot?.p95 ?? 0), width: 20, thresholds: [50, 80] })] }), _jsxs(Box, { children: [_jsx(Text, { children: 'p99'.padEnd(6) }), _jsx(Text, { color: snapshot?.p99 ?? 0 > 5 ? 'red' : 'yellow', bold: true, children: formatMs(snapshot?.p99 ?? 0).padEnd(10) }), _jsx(MeterBar, { label: "", percent: latencyPercent(snapshot?.p99 ?? 0), width: 20, thresholds: [25, 50] })] }), _jsxs(Box, { children: [_jsx(Text, { children: 'max'.padEnd(6) }), _jsx(Text, { color: "red", bold: true, children: formatMs(snapshot?.max ?? 0).padEnd(10) }), _jsx(MeterBar, { label: "", percent: latencyPercent(snapshot?.max ?? 0), width: 20, thresholds: [25, 50] })] })] }), _jsx(Box, { marginTop: 1 }), _jsx(Text, { bold: true, children: " Throughput" }), _jsx(StatusLine, { status: "ok", label: `Success ${formatNum(snapshot?.successCount ?? 0)}`, detail: `${successRate}%` }), _jsx(StatusLine, { status: snapshot?.errorCount ? 'fail' : 'ok', label: `Errors ${formatNum(snapshot?.errorCount ?? 0)}`, detail: `${errorRate}%` }), _jsx(StatusLine, { status: snapshot?.timeoutCount ? 'warn' : 'info', label: `Timeouts ${formatNum(snapshot?.timeoutCount ?? 0)}` }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: `ℹ ${config.rate} req/s target | ${config.connections} connections` }) }), phase === 'stopped' ? (_jsx(Text, { color: "yellow", children: "Stopped early. Waiting for results..." })) : (_jsx(Text, { dimColor: true, children: "q Stop early" }))] }))] }) }));
70
+ }
71
+ //# sourceMappingURL=LoadTestRunView.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestRunView.js","sourceRoot":"","sources":["../../../src/ui/views/LoadTestRunView.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { LoadTestResults } from '../../lib/loadtest-stats.js';
2
+ interface LoadTestSummaryViewProps {
3
+ results: LoadTestResults;
4
+ onRerun?: () => void;
5
+ onBack?: () => void;
6
+ }
7
+ export declare function LoadTestSummaryView({ results, onRerun, onBack }: LoadTestSummaryViewProps): import("react/jsx-runtime").JSX.Element;
8
+ export {};
9
+ //# sourceMappingURL=LoadTestSummaryView.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestSummaryView.d.ts","sourceRoot":"","sources":["../../../src/ui/views/LoadTestSummaryView.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,6BAA6B,CAAC;AAEnE,UAAU,wBAAwB;IAChC,OAAO,EAAE,eAAe,CAAC;IACzB,OAAO,CAAC,EAAE,MAAM,IAAI,CAAC;IACrB,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;CACrB;AAmBD,wBAAgB,mBAAmB,CAAC,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE,wBAAwB,2CAyEzF"}
@@ -0,0 +1,72 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { Box, Text, useInput } from 'ink';
3
+ import { AppShell, Section, DataTable, StatusBox } from '../components/index.js';
4
+ function fmt(n) {
5
+ return n.toLocaleString('en-US');
6
+ }
7
+ function fmtMs(ms) {
8
+ if (ms < 1)
9
+ return `${(ms * 1000).toFixed(0)}μs`;
10
+ if (ms < 1000)
11
+ return `${ms.toFixed(1)}ms`;
12
+ return `${(ms / 1000).toFixed(2)}s`;
13
+ }
14
+ function fmtBytes(bytes) {
15
+ if (bytes >= 1_073_741_824)
16
+ return `${(bytes / 1_073_741_824).toFixed(1)} GB`;
17
+ if (bytes >= 1_048_576)
18
+ return `${(bytes / 1_048_576).toFixed(1)} MB`;
19
+ if (bytes >= 1024)
20
+ return `${(bytes / 1024).toFixed(1)} KB`;
21
+ return `${bytes} B`;
22
+ }
23
+ export function LoadTestSummaryView({ results, onRerun, onBack }) {
24
+ useInput((input, key) => {
25
+ if (input === 'r' && onRerun)
26
+ onRerun();
27
+ if (input === 'q' || key.escape)
28
+ onBack?.();
29
+ });
30
+ const totalReqs = results.totalRequests;
31
+ const successPct = totalReqs > 0 ? ((results.successCount / totalReqs) * 100).toFixed(1) : '0.0';
32
+ const errorPct = totalReqs > 0 ? ((results.errorCount / totalReqs) * 100).toFixed(1) : '0.0';
33
+ const avgRate = results.duration > 0 ? Math.round(totalReqs / (results.duration / 1000)) : 0;
34
+ const rows = [
35
+ { metric: 'Duration', value: `${(results.duration / 1000).toFixed(1)}s` },
36
+ { metric: 'Total Requests', value: fmt(totalReqs) },
37
+ { metric: 'Avg Rate', value: `${fmt(avgRate)} req/s` },
38
+ { metric: 'Success', value: `${fmt(results.successCount)} (${successPct}%)` },
39
+ { metric: 'Errors', value: `${fmt(results.errorCount)} (${errorPct}%)` },
40
+ { metric: 'Timeouts', value: fmt(results.timeoutCount) },
41
+ { metric: 'p50 Latency', value: fmtMs(results.latency.p50) },
42
+ { metric: 'p95 Latency', value: fmtMs(results.latency.p95) },
43
+ { metric: 'p99 Latency', value: fmtMs(results.latency.p99) },
44
+ { metric: 'Max Latency', value: fmtMs(results.latency.max) },
45
+ { metric: 'Std Dev', value: fmtMs(results.latency.stddev) },
46
+ { metric: 'Data Received', value: fmtBytes(results.bytesRead) },
47
+ ];
48
+ // Assessments
49
+ const assessments = [];
50
+ if (results.latency.p99 <= 5) {
51
+ assessments.push({ variant: 'success', text: `PASS: p99 ${fmtMs(results.latency.p99)} < 5ms target` });
52
+ }
53
+ else {
54
+ assessments.push({ variant: 'error', text: `FAIL: p99 ${fmtMs(results.latency.p99)} exceeds 5ms target` });
55
+ }
56
+ const errorRateNum = totalReqs > 0 ? (results.errorCount / totalReqs) * 100 : 0;
57
+ if (errorRateNum <= 1) {
58
+ assessments.push({ variant: 'success', text: `PASS: Error rate ${errorRateNum.toFixed(1)}% within 1% threshold` });
59
+ }
60
+ else {
61
+ assessments.push({ variant: 'warning', text: `WARN: Error rate ${errorRateNum.toFixed(1)}% exceeds 1% threshold` });
62
+ }
63
+ const overallVariant = assessments.every((a) => a.variant === 'success') ? 'success' : 'warning';
64
+ return (_jsx(AppShell, { command: "ct loadtest", onBack: onBack, isComplete: true, children: _jsx(Section, { title: `Load Test Complete — ${results.config.policyName}`, children: _jsxs(Box, { paddingLeft: 2, flexDirection: "column", children: [_jsx(DataTable, { columns: [
65
+ { key: 'metric', label: 'Metric', width: 16 },
66
+ { key: 'value', label: 'Value', width: 24 },
67
+ ], rows: rows }), _jsx(Box, { marginTop: 1 }), _jsx(StatusBox, { variant: overallVariant, title: overallVariant === 'success' ? 'All checks passed' : 'Some checks need attention', lines: assessments.map((a) => {
68
+ const icon = a.variant === 'success' ? '✔' : a.variant === 'warning' ? '⚠' : '✘';
69
+ return `${icon} ${a.text}`;
70
+ }) }), _jsx(Box, { marginTop: 1, children: _jsx(Text, { dimColor: true, children: 'r Re-run q Back' }) })] }) }) }));
71
+ }
72
+ //# sourceMappingURL=LoadTestSummaryView.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestSummaryView.js","sourceRoot":"","sources":["../../../src/ui/views/LoadTestSummaryView.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ import type { LoadTestCliOptions } from './LoadTestConfigView.js';
2
+ interface LoadTestViewProps {
3
+ onBack?: () => void;
4
+ cliOptions?: LoadTestCliOptions;
5
+ }
6
+ export declare function LoadTestView({ onBack, cliOptions }: LoadTestViewProps): import("react/jsx-runtime").JSX.Element | null;
7
+ export {};
8
+ //# sourceMappingURL=LoadTestView.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestView.d.ts","sourceRoot":"","sources":["../../../src/ui/views/LoadTestView.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAsB,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AAMtF,UAAU,iBAAiB;IACzB,MAAM,CAAC,EAAE,MAAM,IAAI,CAAC;IACpB,UAAU,CAAC,EAAE,kBAAkB,CAAC;CACjC;AAED,wBAAgB,YAAY,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,EAAE,iBAAiB,kDA4DrE"}
@@ -0,0 +1,43 @@
1
+ import { jsx as _jsx } from "react/jsx-runtime";
2
+ import { useState, useCallback } from 'react';
3
+ import { LoadTestConfigView } from './LoadTestConfigView.js';
4
+ import { LoadTestRunView } from './LoadTestRunView.js';
5
+ import { LoadTestSummaryView } from './LoadTestSummaryView.js';
6
+ import { useApp } from 'ink';
7
+ export function LoadTestView({ onBack, cliOptions }) {
8
+ const { exit } = useApp();
9
+ const [phase, setPhase] = useState('config');
10
+ const [userConfig, setUserConfig] = useState(null);
11
+ const [results, setResults] = useState(null);
12
+ const handleStart = useCallback((config) => {
13
+ setUserConfig(config);
14
+ setPhase('running');
15
+ }, []);
16
+ const handleComplete = useCallback((res) => {
17
+ setResults(res);
18
+ setPhase('summary');
19
+ }, []);
20
+ const handleRerun = useCallback(() => {
21
+ setResults(null);
22
+ setPhase('config');
23
+ }, []);
24
+ const handleBack = useCallback(() => {
25
+ if (onBack) {
26
+ onBack();
27
+ }
28
+ else {
29
+ exit();
30
+ }
31
+ }, [onBack, exit]);
32
+ if (phase === 'config') {
33
+ return (_jsx(LoadTestConfigView, { onBack: handleBack, onStart: handleStart, cliOptions: cliOptions }));
34
+ }
35
+ if (phase === 'running' && userConfig) {
36
+ return (_jsx(LoadTestRunView, { userConfig: userConfig, onComplete: handleComplete, onBack: handleBack }));
37
+ }
38
+ if (phase === 'summary' && results) {
39
+ return (_jsx(LoadTestSummaryView, { results: results, onRerun: handleRerun, onBack: handleBack }));
40
+ }
41
+ return null;
42
+ }
43
+ //# sourceMappingURL=LoadTestView.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"LoadTestView.js","sourceRoot":"","sources":["../../../src/ui/views/LoadTestView.tsx"],"names":[],"mappings":";AAAA,OAAc,EAAE,QAAQ,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AACrD,OAAO,EAAE,kBAAkB,EAAE,MAAM,yBAAyB,CAAC;AAC7D,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AACvD,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAG/D,OAAO,EAAE,MAAM,EAAE,MAAM,KAAK,CAAC;AAS7B,MAAM,UAAU,YAAY,CAAC,EAAE,MAAM,EAAE,UAAU,EAAqB;IACpE,MAAM,EAAE,IAAI,EAAE,GAAG,MAAM,EAAE,CAAC;IAC1B,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAQ,QAAQ,CAAC,CAAC;IACpD,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,GAAG,QAAQ,CAA4B,IAAI,CAAC,CAAC;IAC9E,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,GAAG,QAAQ,CAAyB,IAAI,CAAC,CAAC;IAErE,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,MAA0B,EAAE,EAAE;QAC7D,aAAa,CAAC,MAAM,CAAC,CAAC;QACtB,QAAQ,CAAC,SAAS,CAAC,CAAC;IACtB,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,MAAM,cAAc,GAAG,WAAW,CAAC,CAAC,GAAoB,EAAE,EAAE;QAC1D,UAAU,CAAC,GAAG,CAAC,CAAC;QAChB,QAAQ,CAAC,SAAS,CAAC,CAAC;IACtB,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,MAAM,WAAW,GAAG,WAAW,CAAC,GAAG,EAAE;QACnC,UAAU,CAAC,IAAI,CAAC,CAAC;QACjB,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACrB,CAAC,EAAE,EAAE,CAAC,CAAC;IAEP,MAAM,UAAU,GAAG,WAAW,CAAC,GAAG,EAAE;QAClC,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,EAAE,CAAC;QACX,CAAC;aAAM,CAAC;YACN,IAAI,EAAE,CAAC;QACT,CAAC;IACH,CAAC,EAAE,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,CAAC;IAEnB,IAAI,KAAK,KAAK,QAAQ,EAAE,CAAC;QACvB,OAAO,CACL,KAAC,kBAAkB,IACjB,MAAM,EAAE,UAAU,EAClB,OAAO,EAAE,WAAW,EACpB,UAAU,EAAE,UAAU,GACtB,CACH,CAAC;IACJ,CAAC;IAED,IAAI,KAAK,KAAK,SAAS,IAAI,UAAU,EAAE,CAAC;QACtC,OAAO,CACL,KAAC,eAAe,IACd,UAAU,EAAE,UAAU,EACtB,UAAU,EAAE,cAAc,EAC1B,MAAM,EAAE,UAAU,GAClB,CACH,CAAC;IACJ,CAAC;IAED,IAAI,KAAK,KAAK,SAAS,IAAI,OAAO,EAAE,CAAC;QACnC,OAAO,CACL,KAAC,mBAAmB,IAClB,OAAO,EAAE,OAAO,EAChB,OAAO,EAAE,WAAW,EACpB,MAAM,EAAE,UAAU,GAClB,CACH,CAAC;IACJ,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"MainMenu.d.ts","sourceRoot":"","sources":["../../../src/ui/views/MainMenu.tsx"],"names":[],"mappings":"AAgNA,UAAU,aAAa;IACrB,OAAO,EAAE,MAAM,CAAC;IAChB,yEAAyE;IACzE,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,2EAA2E;IAC3E,YAAY,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;IACxC,MAAM,EAAE,MAAM,IAAI,CAAC;CACpB;AAED,wBAAgB,QAAQ,CAAC,EAAE,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,MAAM,EAAE,EAAE,aAAa,2CAuZlF"}
1
+ {"version":3,"file":"MainMenu.d.ts","sourceRoot":"","sources":["../../../src/ui/views/MainMenu.tsx"],"names":[],"mappings":"AAgNA,UAAU,aAAa;IACrB,OAAO,EAAE,MAAM,CAAC;IAChB,yEAAyE;IACzE,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,2EAA2E;IAC3E,YAAY,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC;IACxC,MAAM,EAAE,MAAM,IAAI,CAAC;CACpB;AAED,wBAAgB,QAAQ,CAAC,EAAE,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,MAAM,EAAE,EAAE,aAAa,2CAmdlF"}
@@ -320,27 +320,82 @@ export function MainMenu({ version, firstRun, onCliCommand, onExit }) {
320
320
  return;
321
321
  }
322
322
  if (focusZone === 'dashboard') {
323
- if (key.upArrow) {
324
- if (dashCursor > 0)
325
- setDashCursor(prev => prev - 1);
326
- }
327
- else if (key.downArrow) {
328
- if (dashCursor < dashItems.length - 1) {
329
- setDashCursor(prev => prev + 1);
323
+ const inHealthGrid = dashCursor < healthItemCount;
324
+ if (inHealthGrid && healthItemCount > 1) {
325
+ // 2D grid navigation for health items (2 columns)
326
+ const col = dashCursor % 2;
327
+ const row = Math.floor(dashCursor / 2);
328
+ const maxRow = Math.floor((healthItemCount - 1) / 2);
329
+ if (key.upArrow) {
330
+ if (row > 0)
331
+ setDashCursor(prev => prev - 2);
330
332
  }
331
- else {
333
+ else if (key.downArrow) {
334
+ if (row < maxRow) {
335
+ setDashCursor(prev => Math.min(prev + 2, healthItemCount - 1));
336
+ }
337
+ else if (dashItems.length > healthItemCount) {
338
+ setDashCursor(healthItemCount);
339
+ }
340
+ else {
341
+ setFocusZone('menu');
342
+ setMenuCursor(0);
343
+ }
344
+ }
345
+ else if (key.leftArrow) {
346
+ if (col > 0)
347
+ setDashCursor(prev => prev - 1);
348
+ }
349
+ else if (key.rightArrow) {
350
+ if (col === 0 && dashCursor + 1 < healthItemCount) {
351
+ setDashCursor(prev => prev + 1);
352
+ }
353
+ else {
354
+ const item = dashItems[dashCursor];
355
+ if (item)
356
+ openDashItem(item);
357
+ }
358
+ }
359
+ else if (key.return) {
360
+ const item = dashItems[dashCursor];
361
+ if (item)
362
+ openDashItem(item);
363
+ }
364
+ else if (key.escape || input === 'q') {
332
365
  setFocusZone('menu');
333
366
  setMenuCursor(0);
334
367
  }
335
368
  }
336
- else if (key.return || key.rightArrow) {
337
- const item = dashItems[dashCursor];
338
- if (item)
339
- openDashItem(item);
340
- }
341
- else if (key.escape || input === 'q') {
342
- setFocusZone('menu');
343
- setMenuCursor(0);
369
+ else {
370
+ // Linear navigation for container items (or single health item)
371
+ if (key.upArrow) {
372
+ if (dashCursor > healthItemCount) {
373
+ setDashCursor(prev => prev - 1);
374
+ }
375
+ else if (healthItemCount > 0) {
376
+ // Jump back into health grid, last row left column
377
+ const lastRow = Math.floor((healthItemCount - 1) / 2);
378
+ setDashCursor(lastRow * 2);
379
+ }
380
+ }
381
+ else if (key.downArrow) {
382
+ if (dashCursor < dashItems.length - 1) {
383
+ setDashCursor(prev => prev + 1);
384
+ }
385
+ else {
386
+ setFocusZone('menu');
387
+ setMenuCursor(0);
388
+ }
389
+ }
390
+ else if (key.return || key.rightArrow) {
391
+ const item = dashItems[dashCursor];
392
+ if (item)
393
+ openDashItem(item);
394
+ }
395
+ else if (key.escape || input === 'q') {
396
+ setFocusZone('menu');
397
+ setMenuCursor(0);
398
+ }
344
399
  }
345
400
  return;
346
401
  }
@@ -422,11 +477,18 @@ export function MainMenu({ version, firstRun, onCliCommand, onExit }) {
422
477
  const webSvc = serviceList.find(s => s.service === 'web');
423
478
  const uptime = webSvc ? extractUptime(webSvc.status) : '';
424
479
  return uptime && uptime !== 'exited' ? (_jsxs(Text, { dimColor: true, children: [' ', "Up ", uptime] })) : null;
425
- })()] }), dashItems.slice(0, healthItemCount).map((item, i) => {
426
- const isFocused = focusZone === 'dashboard' && dashCursor === i;
427
- const prefix = isFocused ? '❯' : ' ';
428
- return (_jsxs(Box, { children: [_jsx(Text, { color: isFocused ? 'cyan' : undefined, bold: isFocused, children: `${prefix} ` }), _jsx(Text, { color: item.color, children: item.icon }), _jsx(Text, { color: isFocused ? 'cyan' : undefined, bold: isFocused, children: ` ${item.label.padEnd(10)}` }), _jsx(Text, { color: item.color, children: item.status.padEnd(16) }), isFocused && _jsx(Text, { dimColor: true, children: item.hint })] }, `health-${i}`));
429
- })] })), certDaysLeft !== null && (_jsx(Box, { paddingLeft: 2, marginBottom: 1, children: _jsx(Text, { color: "yellow", children: `\u26A0 Certificate expires in ${certDaysLeft} day${certDaysLeft !== 1 ? 's' : ''}` }) })), !services && (_jsx(Box, { paddingLeft: 2, marginBottom: 1, children: _jsx(Spinner, { label: "Loading containers..." }) })), serviceList.length > 0 && (_jsxs(Box, { flexDirection: "column", paddingLeft: 2, marginBottom: 1, children: [_jsxs(Box, { marginTop: 1, children: [_jsx(Text, { bold: true, children: "Containers" }), _jsxs(Text, { dimColor: true, children: [' ', serviceList.length, " total"] }), healthyCount > 0 && _jsxs(Text, { color: "green", children: [" \u2014 ", healthyCount, " healthy"] }), unhealthyCount > 0 && _jsxs(Text, { color: "yellow", children: [", ", unhealthyCount, " unhealthy"] }), stoppedCount > 0 && _jsxs(Text, { color: "red", children: [", ", stoppedCount, " stopped"] })] }), dashItems.slice(healthItemCount).map((item, i) => {
480
+ })()] }), Array.from({ length: Math.ceil(healthItemCount / 2) }, (_, rowIdx) => {
481
+ const leftIdx = rowIdx * 2;
482
+ const rightIdx = rowIdx * 2 + 1;
483
+ const leftItem = dashItems[leftIdx];
484
+ const rightItem = rightIdx < healthItemCount ? dashItems[rightIdx] : null;
485
+ const renderHealthCell = (item, idx) => {
486
+ const isFocused = focusZone === 'dashboard' && dashCursor === idx;
487
+ const prefix = isFocused ? '❯' : ' ';
488
+ return (_jsxs(Box, { width: "50%", children: [_jsx(Text, { color: isFocused ? 'cyan' : undefined, bold: isFocused, children: `${prefix} ` }), _jsx(Text, { color: item.color, children: item.icon }), _jsx(Text, { color: isFocused ? 'cyan' : undefined, bold: isFocused, children: ` ${item.label.padEnd(10)}` }), _jsx(Text, { color: item.color, children: item.status })] }));
489
+ };
490
+ return (_jsxs(Box, { flexDirection: "row", children: [leftItem && renderHealthCell(leftItem, leftIdx), rightItem && renderHealthCell(rightItem, rightIdx)] }, `health-row-${rowIdx}`));
491
+ }), focusZone === 'dashboard' && dashCursor < healthItemCount && dashItems[dashCursor] && (_jsx(Box, { paddingLeft: 2, children: _jsx(Text, { dimColor: true, children: dashItems[dashCursor].hint }) }))] })), certDaysLeft !== null && (_jsx(Box, { paddingLeft: 2, marginBottom: 1, children: _jsx(Text, { color: "yellow", children: `\u26A0 Certificate expires in ${certDaysLeft} day${certDaysLeft !== 1 ? 's' : ''}` }) })), !services && (_jsx(Box, { paddingLeft: 2, marginBottom: 1, children: _jsx(Spinner, { label: "Loading containers..." }) })), serviceList.length > 0 && (_jsxs(Box, { flexDirection: "column", paddingLeft: 2, marginBottom: 1, children: [_jsxs(Box, { marginTop: 1, children: [_jsx(Text, { bold: true, children: "Containers" }), _jsxs(Text, { dimColor: true, children: [' ', serviceList.length, " total"] }), healthyCount > 0 && _jsxs(Text, { color: "green", children: [" \u2014 ", healthyCount, " healthy"] }), unhealthyCount > 0 && _jsxs(Text, { color: "yellow", children: [", ", unhealthyCount, " unhealthy"] }), stoppedCount > 0 && _jsxs(Text, { color: "red", children: [", ", stoppedCount, " stopped"] })] }), dashItems.slice(healthItemCount).map((item, i) => {
430
492
  const globalIdx = healthItemCount + i;
431
493
  const isFocused = focusZone === 'dashboard' && dashCursor === globalIdx;
432
494
  const prefix = isFocused ? '❯' : ' ';