Utku Ozdemir a7b79ef1be
feat: add network config screen to dashboard
Implement the network config screen with input forms to configure the initial node networking by writing a config to the META partition.

Closes siderolabs/talos#6961.

Signed-off-by: Utku Ozdemir <utku.ozdemir@siderolabs.com>
2023-03-23 17:29:52 +04:00

210 lines
5.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 resourcedata implements the types and the data sources for the data sourced from the Talos resource API (COSI).
package resourcedata
import (
"context"
"errors"
"fmt"
"sync"
"time"
"github.com/cosi-project/runtime/pkg/resource"
"github.com/cosi-project/runtime/pkg/state"
"github.com/siderolabs/gen/channel"
"golang.org/x/sync/errgroup"
"google.golang.org/grpc/metadata"
"github.com/siderolabs/talos/internal/pkg/meta"
"github.com/siderolabs/talos/pkg/machinery/client"
"github.com/siderolabs/talos/pkg/machinery/resources/cluster"
"github.com/siderolabs/talos/pkg/machinery/resources/config"
"github.com/siderolabs/talos/pkg/machinery/resources/hardware"
"github.com/siderolabs/talos/pkg/machinery/resources/k8s"
"github.com/siderolabs/talos/pkg/machinery/resources/network"
"github.com/siderolabs/talos/pkg/machinery/resources/runtime"
)
// Data contains a resource, whether it is deleted and the node it came from.
type Data struct {
Node string
Resource resource.Resource
Deleted bool
}
// Source is the data source for the Talos resources.
type Source struct {
ctxCancel context.CancelFunc
eg errgroup.Group
once sync.Once
COSI state.State
ch chan Data
NodeResourceCh <-chan Data
}
// Run starts the data source.
func (source *Source) Run(ctx context.Context) {
source.once.Do(func() {
source.run(ctx)
})
}
// Stop stops the data source.
func (source *Source) Stop() error {
source.ctxCancel()
return source.eg.Wait()
}
func (source *Source) run(ctx context.Context) {
ctx, source.ctxCancel = context.WithCancel(ctx)
source.ch = make(chan Data)
source.NodeResourceCh = source.ch
nodes := source.nodes(ctx)
for _, node := range nodes {
node := node
source.eg.Go(func() error {
source.runResourceWatchWithRetries(ctx, node)
return nil
})
}
}
func (source *Source) runResourceWatchWithRetries(ctx context.Context, node string) {
for {
if err := source.runResourceWatch(ctx, node); errors.Is(err, context.Canceled) {
return
}
// wait for a second before the next retry
timer := time.NewTimer(1 * time.Second)
select {
case <-ctx.Done():
timer.Stop()
return
case <-timer.C:
}
}
}
//nolint:gocyclo,cyclop
func (source *Source) runResourceWatch(ctx context.Context, node string) error {
if node != "" {
ctx = client.WithNode(ctx, node)
}
ctx, cancel := context.WithCancel(ctx)
defer cancel()
eventCh := make(chan state.Event)
if err := source.COSI.Watch(ctx, runtime.NewMachineStatus().Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, config.NewMachineType().Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, k8s.NewKubeletSpec(k8s.NamespaceName, k8s.KubeletID).Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, network.NewResolverStatus(network.NamespaceName, network.ResolverID).Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, network.NewTimeServerStatus(network.NamespaceName, network.TimeServerID).Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, hardware.NewSystemInformation(hardware.SystemInformationID).Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, cluster.NewInfo().Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.Watch(ctx, runtime.NewMetaKey(runtime.NamespaceName, runtime.MetaKeyTagToID(meta.MetalNetworkPlatformConfig)).Metadata(), eventCh); err != nil {
return err
}
if err := source.COSI.WatchKind(ctx, k8s.NewStaticPodStatus(k8s.NamespaceName, "").Metadata(), eventCh, state.WithBootstrapContents(true)); err != nil {
return err
}
if err := source.COSI.WatchKind(ctx, network.NewRouteStatus(network.NamespaceName, "").Metadata(), eventCh, state.WithBootstrapContents(true)); err != nil {
return err
}
if err := source.COSI.WatchKind(ctx, network.NewLinkStatus(network.NamespaceName, "").Metadata(), eventCh, state.WithBootstrapContents(true)); err != nil {
return err
}
if err := source.COSI.WatchKind(ctx, cluster.NewMember(cluster.NamespaceName, "").Metadata(), eventCh, state.WithBootstrapContents(true)); err != nil {
return err
}
for {
select {
case <-ctx.Done():
return ctx.Err()
case event := <-eventCh:
switch event.Type {
case state.Errored:
return fmt.Errorf("watch failed: %w", event.Error)
case state.Bootstrapped:
// ignored
case state.Created, state.Updated:
if !channel.SendWithContext(ctx, source.ch, Data{
Node: node,
Resource: event.Resource,
}) {
return ctx.Err()
}
case state.Destroyed:
if !channel.SendWithContext(ctx, source.ch, Data{
Node: node,
Resource: event.Resource,
Deleted: true,
}) {
return ctx.Err()
}
}
}
}
}
func (source *Source) nodes(ctx context.Context) []string {
md, mdOk := metadata.FromOutgoingContext(ctx)
if !mdOk {
return []string{""} // local node
}
nodeVal := md.Get("node")
if len(nodeVal) > 0 {
return []string{nodeVal[0]}
}
nodesVal := md.Get("nodes")
if len(nodesVal) == 0 {
return []string{""} // local node
}
return nodesVal
}