@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.
- package/dist/commands/loadtest.d.ts +3 -0
- package/dist/commands/loadtest.d.ts.map +1 -0
- package/dist/commands/loadtest.js +22 -0
- package/dist/commands/loadtest.js.map +1 -0
- package/dist/index.js +3 -1
- package/dist/index.js.map +1 -1
- package/dist/lib/curri-xml.d.ts +21 -0
- package/dist/lib/curri-xml.d.ts.map +1 -0
- package/dist/lib/curri-xml.js +53 -0
- package/dist/lib/curri-xml.js.map +1 -0
- package/dist/lib/loadtest-stats.d.ts +91 -0
- package/dist/lib/loadtest-stats.d.ts.map +1 -0
- package/dist/lib/loadtest-stats.js +23 -0
- package/dist/lib/loadtest-stats.js.map +1 -0
- package/dist/lib/loadtest.d.ts +14 -0
- package/dist/lib/loadtest.d.ts.map +1 -0
- package/dist/lib/loadtest.js +112 -0
- package/dist/lib/loadtest.js.map +1 -0
- package/dist/lib/rpc-scripts.d.ts +2 -0
- package/dist/lib/rpc-scripts.d.ts.map +1 -1
- package/dist/lib/rpc-scripts.js +29 -0
- package/dist/lib/rpc-scripts.js.map +1 -1
- package/dist/ui/components/Sparkline.d.ts +15 -0
- package/dist/ui/components/Sparkline.d.ts.map +1 -0
- package/dist/ui/components/Sparkline.js +21 -0
- package/dist/ui/components/Sparkline.js.map +1 -0
- package/dist/ui/components/index.d.ts +1 -0
- package/dist/ui/components/index.d.ts.map +1 -1
- package/dist/ui/components/index.js +1 -0
- package/dist/ui/components/index.js.map +1 -1
- package/dist/ui/hooks/index.d.ts +2 -0
- package/dist/ui/hooks/index.d.ts.map +1 -1
- package/dist/ui/hooks/index.js +1 -0
- package/dist/ui/hooks/index.js.map +1 -1
- package/dist/ui/hooks/useLoadTest.d.ts +11 -0
- package/dist/ui/hooks/useLoadTest.d.ts.map +1 -0
- package/dist/ui/hooks/useLoadTest.js +38 -0
- package/dist/ui/hooks/useLoadTest.js.map +1 -0
- package/dist/ui/views/LoadTestConfigView.d.ts +31 -0
- package/dist/ui/views/LoadTestConfigView.d.ts.map +1 -0
- package/dist/ui/views/LoadTestConfigView.js +111 -0
- package/dist/ui/views/LoadTestConfigView.js.map +1 -0
- package/dist/ui/views/LoadTestRunView.d.ts +10 -0
- package/dist/ui/views/LoadTestRunView.d.ts.map +1 -0
- package/dist/ui/views/LoadTestRunView.js +71 -0
- package/dist/ui/views/LoadTestRunView.js.map +1 -0
- package/dist/ui/views/LoadTestSummaryView.d.ts +9 -0
- package/dist/ui/views/LoadTestSummaryView.d.ts.map +1 -0
- package/dist/ui/views/LoadTestSummaryView.js +72 -0
- package/dist/ui/views/LoadTestSummaryView.js.map +1 -0
- package/dist/ui/views/LoadTestView.d.ts +8 -0
- package/dist/ui/views/LoadTestView.d.ts.map +1 -0
- package/dist/ui/views/LoadTestView.js +43 -0
- package/dist/ui/views/LoadTestView.js.map +1 -0
- package/dist/ui/views/MainMenu.d.ts.map +1 -1
- package/dist/ui/views/MainMenu.js +83 -21
- package/dist/ui/views/MainMenu.js.map +1 -1
- package/dist/ui/views/StatusView.d.ts.map +1 -1
- package/dist/ui/views/StatusView.js +134 -21
- package/dist/ui/views/StatusView.js.map +1 -1
- 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,
|
|
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
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
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
|
|
337
|
-
|
|
338
|
-
if (
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
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
|
-
})()] }),
|
|
426
|
-
const
|
|
427
|
-
const
|
|
428
|
-
|
|
429
|
-
|
|
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 ? '❯' : ' ';
|