BMC: - collector/board.go: collectBMCFirmware() via ipmitool mc info, graceful skip if /dev/ipmi0 absent - collector/collector.go: append BMC firmware record to snap.Firmware - app/panel.go: show BMC version in TUI right-panel header alongside BIOS CPU SAT: - platform/sat.go: RunCPUAcceptancePack(baseDir, durationSec) — lscpu + sensors before/after + stress-ng - app/app.go: RunCPUAcceptancePack + RunCPUAcceptancePackResult methods, satRunner interface updated - app/panel.go: CPU row now reads real PASS/FAIL from cpu-*/summary.txt via satStatuses(); cpuDetailResult shows last SAT summary + audit data - tui/types.go: actionRunCPUSAT, confirmBody for CPU test with mode label - tui/screen_health_check.go: hcCPUDurations [60,300,900]s; hcRunSingle(CPU)→confirm screen; executeRunAll uses RunCPUAcceptancePackResult - tui/forms.go: actionRunCPUSAT → RunCPUAcceptancePackResult with mode duration - cmd/bee/main.go: bee sat cpu [--duration N] subcommand Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
119 lines
3.1 KiB
Go
119 lines
3.1 KiB
Go
package tui
|
|
|
|
import tea "github.com/charmbracelet/bubbletea"
|
|
|
|
func (m model) updateStaticForm(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
|
switch msg.String() {
|
|
case "esc":
|
|
m.screen = screenNetwork
|
|
m.formFields = nil
|
|
m.formIndex = 0
|
|
return m, nil
|
|
case "up", "shift+tab":
|
|
if m.formIndex > 0 {
|
|
m.formIndex--
|
|
}
|
|
case "down", "tab":
|
|
if m.formIndex < len(m.formFields)-1 {
|
|
m.formIndex++
|
|
}
|
|
case "enter":
|
|
if m.formIndex < len(m.formFields)-1 {
|
|
m.formIndex++
|
|
return m, nil
|
|
}
|
|
cfg := m.app.ParseStaticIPv4Config(m.selectedIface, []string{
|
|
m.formFields[0].Value,
|
|
m.formFields[1].Value,
|
|
m.formFields[2].Value,
|
|
m.formFields[3].Value,
|
|
})
|
|
m.busy = true
|
|
m.busyTitle = "Static IPv4: " + m.selectedIface
|
|
return m, func() tea.Msg {
|
|
result, err := m.app.SetStaticIPv4Result(cfg)
|
|
return resultMsg{title: result.Title, body: result.Body, err: err, back: screenNetwork}
|
|
}
|
|
case "backspace":
|
|
field := &m.formFields[m.formIndex]
|
|
if len(field.Value) > 0 {
|
|
field.Value = field.Value[:len(field.Value)-1]
|
|
}
|
|
default:
|
|
if msg.Type == tea.KeyRunes && len(msg.Runes) > 0 {
|
|
m.formFields[m.formIndex].Value += string(msg.Runes)
|
|
}
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func (m model) updateConfirm(msg tea.KeyMsg) (tea.Model, tea.Cmd) {
|
|
switch msg.String() {
|
|
case "left", "up", "tab":
|
|
if m.cursor > 0 {
|
|
m.cursor--
|
|
}
|
|
case "right", "down":
|
|
if m.cursor < 1 {
|
|
m.cursor++
|
|
}
|
|
case "esc":
|
|
m.screen = m.confirmCancelTarget()
|
|
m.cursor = 0
|
|
m.pendingAction = actionNone
|
|
return m, nil
|
|
case "enter":
|
|
if m.cursor == 1 { // Cancel
|
|
m.screen = m.confirmCancelTarget()
|
|
m.cursor = 0
|
|
m.pendingAction = actionNone
|
|
return m, nil
|
|
}
|
|
m.busy = true
|
|
switch m.pendingAction {
|
|
case actionExportBundle:
|
|
m.busyTitle = "Export support bundle"
|
|
target := *m.selectedTarget
|
|
return m, func() tea.Msg {
|
|
result, err := m.app.ExportSupportBundleResult(target)
|
|
return resultMsg{title: result.Title, body: result.Body, err: err, back: screenMain}
|
|
}
|
|
case actionRunAll:
|
|
return m.executeRunAll()
|
|
case actionRunMemorySAT:
|
|
m.busyTitle = "Memory test"
|
|
return m, func() tea.Msg {
|
|
result, err := m.app.RunMemoryAcceptancePackResult("")
|
|
return resultMsg{title: result.Title, body: result.Body, err: err, back: screenHealthCheck}
|
|
}
|
|
case actionRunStorageSAT:
|
|
m.busyTitle = "Storage test"
|
|
return m, func() tea.Msg {
|
|
result, err := m.app.RunStorageAcceptancePackResult("")
|
|
return resultMsg{title: result.Title, body: result.Body, err: err, back: screenHealthCheck}
|
|
}
|
|
case actionRunCPUSAT:
|
|
m.busyTitle = "CPU test"
|
|
durationSec := hcCPUDurations[m.hcMode]
|
|
return m, func() tea.Msg {
|
|
result, err := m.app.RunCPUAcceptancePackResult("", durationSec)
|
|
return resultMsg{title: result.Title, body: result.Body, err: err, back: screenHealthCheck}
|
|
}
|
|
}
|
|
case "ctrl+c":
|
|
return m, tea.Quit
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func (m model) confirmCancelTarget() screen {
|
|
switch m.pendingAction {
|
|
case actionExportBundle:
|
|
return screenExportTargets
|
|
case actionRunAll, actionRunMemorySAT, actionRunStorageSAT, actionRunCPUSAT:
|
|
return screenHealthCheck
|
|
default:
|
|
return screenMain
|
|
}
|
|
}
|