Utku Ozdemir 3735add87c
fix: reconnect to the logs stream in dashboard after reboot
The log stream displayed in the dashboard was stopping to work when a node was rebooted.
Rework the log data source to establish a per-node connection and use a retry loop to always reconnect until the dashboard is terminated.

Print the connection errors in the log stream in red color.

Closes siderolabs/talos#8388.

Signed-off-by: Utku Ozdemir <utku.ozdemir@siderolabs.com>
2024-04-10 10:43:45 +02:00

135 lines
3.3 KiB
Go

// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
package dashboard
import (
"github.com/rivo/tview"
"github.com/siderolabs/talos/internal/pkg/dashboard/apidata"
"github.com/siderolabs/talos/internal/pkg/dashboard/components"
"github.com/siderolabs/talos/internal/pkg/dashboard/resourcedata"
)
// SummaryGrid represents the summary grid with the basic node information and the logs.
type SummaryGrid struct {
tview.Grid
app *tview.Application
apiDataListeners []APIDataListener
resourceListeners []ResourceDataListener
nodeSelectListeners []NodeSelectListener
active bool
node string
logViewers map[string]*components.LogViewer
}
// NewSummaryGrid initializes SummaryGrid.
func NewSummaryGrid(app *tview.Application) *SummaryGrid {
widget := &SummaryGrid{
app: app,
Grid: *tview.NewGrid(),
logViewers: make(map[string]*components.LogViewer),
}
widget.SetRows(8, 0).SetColumns(-3, -2, -3)
talosInfo := components.NewTalosInfo()
widget.AddItem(talosInfo, 0, 0, 1, 1, 0, 0, false)
kubernetesInfo := components.NewKubernetesInfo()
widget.AddItem(kubernetesInfo, 0, 1, 1, 1, 0, 0, false)
networkInfo := components.NewNetworkInfo()
widget.AddItem(networkInfo, 0, 2, 1, 1, 0, 0, false)
widget.apiDataListeners = []APIDataListener{
kubernetesInfo,
}
widget.resourceListeners = []ResourceDataListener{
talosInfo,
kubernetesInfo,
networkInfo,
}
widget.nodeSelectListeners = []NodeSelectListener{
talosInfo,
kubernetesInfo,
networkInfo,
}
return widget
}
// OnNodeSelect implements the NodeSelectListener interface.
func (widget *SummaryGrid) OnNodeSelect(node string) {
widget.node = node
widget.updateLogViewer()
for _, nodeSelectListener := range widget.nodeSelectListeners {
nodeSelectListener.OnNodeSelect(node)
}
}
// OnAPIDataChange implements the APIDataListener interface.
func (widget *SummaryGrid) OnAPIDataChange(node string, data *apidata.Data) {
for _, dataWidget := range widget.apiDataListeners {
dataWidget.OnAPIDataChange(node, data)
}
}
// OnResourceDataChange implements the ResourceDataListener interface.
func (widget *SummaryGrid) OnResourceDataChange(nodeResource resourcedata.Data) {
for _, resourceListener := range widget.resourceListeners {
resourceListener.OnResourceDataChange(nodeResource)
}
}
// OnLogDataChange implements the LogDataListener interface.
func (widget *SummaryGrid) OnLogDataChange(node, logLine, logError string) {
widget.logViewer(node).WriteLog(logLine, logError)
}
func (widget *SummaryGrid) updateLogViewer() {
if !widget.active {
return
}
widget.logViewer(widget.node)
for currNode, logViewer := range widget.logViewers {
if currNode == widget.node {
widget.AddItem(logViewer, 1, 0, 1, 3, 0, 0, false)
widget.app.SetFocus(logViewer)
return
}
widget.RemoveItem(logViewer)
}
}
func (widget *SummaryGrid) logViewer(node string) *components.LogViewer {
logViewer, ok := widget.logViewers[node]
if !ok {
logViewer = components.NewLogViewer()
widget.logViewers[node] = logViewer
}
return logViewer
}
// OnScreenSelect implements the screenSelectListener interface.
func (widget *SummaryGrid) onScreenSelect(active bool) {
widget.active = active
widget.updateLogViewer()
}