Skip to content

Commit

Permalink
Merge pull request #186 from sunya-ch/v1.2.3
Browse files Browse the repository at this point in the history
move to v1.2.3 and add namespace watcher
  • Loading branch information
sunya-ch authored Apr 10, 2024
2 parents dee9650 + b7f185e commit 1dd8ae3
Show file tree
Hide file tree
Showing 11 changed files with 211 additions and 65 deletions.
13 changes: 5 additions & 8 deletions api/v1/hostinterface_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,16 +9,13 @@ import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)

// EDIT THIS FILE! THIS IS SCAFFOLDING FOR YOU TO OWN!
// NOTE: json tags are required. Any new fields you add must have json tags for the fields to be serialized.

type InterfaceInfoType struct {
InterfaceName string `json:"interfaceName"`
NetAddress string `json:"netAddress"`
HostIP string `json:"hostIP"`
Vendor string `json:"vendor"`
Product string `json:"product"`
PciAddress string `json:"pciAddress"`
NetAddress string `json:"netAddress,omitempty"`
HostIP string `json:"hostIP,omitempty"`
Vendor string `json:"vendor,omitempty"`
Product string `json:"product,omitempty"`
PciAddress string `json:"pciAddress,omitempty"`
}

func (i InterfaceInfoType) Equal(cmp InterfaceInfoType) bool {
Expand Down
5 changes: 0 additions & 5 deletions config/crd/bases/multinic.fms.io_hostinterfaces.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -54,12 +54,7 @@ spec:
vendor:
type: string
required:
- hostIP
- interfaceName
- netAddress
- pciAddress
- product
- vendor
type: object
type: array
required:
Expand Down
2 changes: 2 additions & 0 deletions config/manager/manager.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,8 @@ spec:
replicas: 1
template:
metadata:
annotations:
kubectl.kubernetes.io/default-container: manager
labels:
control-plane: controller-manager
spec:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -131,4 +131,4 @@ spec:
provider:
name: Foundation Model Stack
version: 0.0.0
replaces: multi-nic-cni-operator.v1.1.0
replaces: multi-nic-cni-operator.v1.2.0
36 changes: 30 additions & 6 deletions controllers/multinicnetwork_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -132,8 +132,6 @@ func (r *MultiNicNetworkReconciler) Reconcile(ctx context.Context, req ctrl.Requ
instance.Status.DiscoverStatus.ExistDaemon = daemonSize
instance.Status.InterfaceInfoAvailable = infoAvailableSize

// Get main plugin
mainPlugin, annotations, err := r.GetMainPluginConf(instance)
multinicnetworkName := instance.GetName()
if err != nil {
message := fmt.Sprintf("Failed to get main config %s: %v", multinicnetworkName, err)
Expand All @@ -143,10 +141,7 @@ func (r *MultiNicNetworkReconciler) Reconcile(ctx context.Context, req ctrl.Requ
}
vars.NetworkLog.V(2).Info(message)
} else {
mainPlugin = plugin.RemoveEmpty(instance.Spec.MainPlugin.CNIArgs, mainPlugin)
vars.NetworkLog.V(2).Info(fmt.Sprintf("main plugin: %s", mainPlugin))
// Create net attach def
err = r.NetAttachDefHandler.CreateOrUpdate(instance, mainPlugin, annotations)
err = r.GenerateNetAttachDef(instance)
if err != nil {
message := fmt.Sprintf("Failed to create %s: %v", multinicnetworkName, err)
err = r.CIDRHandler.MultiNicNetworkHandler.UpdateNetConfigStatus(instance, multinicv1.ConfigFailed, message)
Expand Down Expand Up @@ -204,6 +199,18 @@ func (r *MultiNicNetworkReconciler) Reconcile(ctx context.Context, req ctrl.Requ
return ctrl.Result{}, nil
}

func (r *MultiNicNetworkReconciler) GenerateNetAttachDef(instance *multinicv1.MultiNicNetwork) error {
// Get main plugin
mainPlugin, annotations, err := r.GetMainPluginConf(instance)
if err == nil {
mainPlugin = plugin.RemoveEmpty(instance.Spec.MainPlugin.CNIArgs, mainPlugin)
vars.NetworkLog.V(2).Info(fmt.Sprintf("main plugin: %s", mainPlugin))
// Create net attach def
err = r.NetAttachDefHandler.CreateOrUpdate(instance, mainPlugin, annotations)
}
return err
}

func (r *MultiNicNetworkReconciler) GetMainPluginConf(instance *multinicv1.MultiNicNetwork) (string, map[string]string, error) {
spec := instance.Spec.MainPlugin
if p, exist := r.PluginMap[spec.Type]; exist {
Expand Down Expand Up @@ -291,3 +298,20 @@ func (r *MultiNicNetworkReconciler) callFinalizer(reqLogger logr.Logger, instanc
r.CIDRHandler.MultiNicNetworkHandler.SafeCache.UnsetCache(instance.Name)
return nil
}

// HandleNewNamespace handles new namespace
// - generate NAD
func (r *MultiNicNetworkReconciler) HandleNewNamespace(namespace string) {
multinicnetworks := r.CIDRHandler.MultiNicNetworkHandler.FilterNetworksByNamespace(namespace)
for _, net := range multinicnetworks {
// Get main plugin
mainPlugin, annotations, err := r.GetMainPluginConf(&net)
if err == nil {
mainPlugin = plugin.RemoveEmpty(net.Spec.MainPlugin.CNIArgs, mainPlugin)
err = r.NetAttachDefHandler.CreateOrUpdateOnNamespace(namespace, &net, mainPlugin, annotations)
}
if err != nil {
vars.NetworkLog.V(2).Info(fmt.Sprintf("Failed to create networkAttachementDefinition for %s on %s: %v", net.Name, namespace, err))
}
}
}
36 changes: 31 additions & 5 deletions controllers/multinicnetwork_handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,7 @@ func (h *MultiNicNetworkHandler) updateStatus(instance *multinicv1.MultiNicNetwo

if !NetStatusUpdated(instance, netStatus) {
vars.NetworkLog.V(2).Info(fmt.Sprintf("No status update %s", instance.Name))
h.SetCache(instance.Name, *instance)
return netStatus, nil
}

Expand All @@ -126,7 +127,7 @@ func (h *MultiNicNetworkHandler) updateStatus(instance *multinicv1.MultiNicNetwo
if err != nil {
vars.NetworkLog.V(2).Info(fmt.Sprintf("Failed to update %s status: %v", instance.Name, err))
} else {
h.SetStatusCache(instance.Name, instance.Status)
h.SetCache(instance.Name, *instance)
}
return netStatus, err
}
Expand Down Expand Up @@ -171,12 +172,12 @@ func (h *MultiNicNetworkHandler) UpdateNetConfigStatus(instance *multinicv1.Mult
if err != nil {
vars.NetworkLog.V(2).Info(fmt.Sprintf("Failed to update %s status: %v", instance.Name, err))
} else {
h.SetStatusCache(instance.Name, instance.Status)
h.SetCache(instance.Name, *instance)
}
return err
}

func (h *MultiNicNetworkHandler) SetStatusCache(key string, value multinicv1.MultiNicNetworkStatus) {
func (h *MultiNicNetworkHandler) SetCache(key string, value multinicv1.MultiNicNetwork) {
h.SafeCache.SetCache(key, value)
}

Expand All @@ -185,15 +186,40 @@ func (h *MultiNicNetworkHandler) GetStatusCache(key string) (multinicv1.MultiNic
if value == nil {
return multinicv1.MultiNicNetworkStatus{}, fmt.Errorf(vars.NotFoundError)
}
return value.(multinicv1.MultiNicNetworkStatus), nil
return value.(multinicv1.MultiNicNetwork).Status, nil
}

func (h *MultiNicNetworkHandler) ListStatusCache() map[string]multinicv1.MultiNicNetworkStatus {
snapshot := make(map[string]multinicv1.MultiNicNetworkStatus)
h.SafeCache.Lock()
for key, value := range h.cache {
snapshot[key] = value.(multinicv1.MultiNicNetworkStatus)
snapshot[key] = value.(multinicv1.MultiNicNetwork).Status
}
h.SafeCache.Unlock()
return snapshot
}

func (h *MultiNicNetworkHandler) FilterNetworksByNamespace(target string) []multinicv1.MultiNicNetwork {
filteredNetworks := []multinicv1.MultiNicNetwork{}
h.SafeCache.Lock()
for _, value := range h.cache {
net := value.(multinicv1.MultiNicNetwork)
namespaces := net.Spec.Namespaces
if len(namespaces) == 0 {
vars.NetworkLog.V(2).Info(fmt.Sprintf("FilterNetworksByNamespace %s has no namespace set", net.Name))
// NOTE: if namespace list is not specified, append
filteredNetworks = append(filteredNetworks, net)
} else {
vars.NetworkLog.V(2).Info(fmt.Sprintf("FilterNetworksByNamespace %s set %v", net.Name, namespaces))
// NOTE: if namespace list is specified and target is in the list, append
for _, ns := range namespaces {
if ns == target {
filteredNetworks = append(filteredNetworks, net)
break
}
}
}
}
h.SafeCache.Unlock()
return filteredNetworks
}
59 changes: 59 additions & 0 deletions controllers/namespace_watcher.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
/*
* Copyright 2022- IBM Inc. All rights reserved
* SPDX-License-Identifier: Apache2.0
*/

package controllers

import (
v1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/util/wait"
"k8s.io/client-go/informers"
"k8s.io/client-go/kubernetes"
"k8s.io/client-go/rest"
"k8s.io/client-go/tools/cache"
"sigs.k8s.io/controller-runtime/pkg/client"
)

// NamespaceWatcher watches new namespace and generate net-attach-def
type NamespaceWatcher struct {
*kubernetes.Clientset
NamespaceQueue chan string
Quit chan struct{}
*MultiNicNetworkReconciler
}

// NewNamespaceWatcher creates new namespace watcher
func NewNamespaceWatcher(client client.Client, config *rest.Config, multinicnetworkReconciler *MultiNicNetworkReconciler, quit chan struct{}) *NamespaceWatcher {
clientset, _ := kubernetes.NewForConfig(config)
watcher := &NamespaceWatcher{
Clientset: clientset,
NamespaceQueue: make(chan string),
Quit: quit,
MultiNicNetworkReconciler: multinicnetworkReconciler,
}
factory := informers.NewSharedInformerFactory(clientset, 0)
nsInformer := factory.Core().V1().Namespaces()

nsInformer.Informer().AddEventHandler(cache.ResourceEventHandlerFuncs{
AddFunc: func(obj interface{}) {
if ns, ok := obj.(*v1.Namespace); ok {
watcher.NamespaceQueue <- ns.Name
}
},
})
factory.Start(watcher.Quit)

return watcher
}

// Run executes namespace watcher routine until get quit signal
func (w *NamespaceWatcher) Run() {
defer close(w.NamespaceQueue)
wait.Until(w.ProcessNamespaceQueue, 0, w.Quit)
}

func (w *NamespaceWatcher) ProcessNamespaceQueue() {
ns := <-w.NamespaceQueue
w.MultiNicNetworkReconciler.HandleNewNamespace(ns)
}
2 changes: 2 additions & 0 deletions controllers/vars/vars.go
Original file line number Diff line number Diff line change
Expand Up @@ -77,6 +77,7 @@ var (

// logger options to change log level on the fly
ZapOpts *zap.Options
SetupLog logr.Logger
DaemonLog logr.Logger
DefLog logr.Logger
CIDRLog logr.Logger
Expand All @@ -103,6 +104,7 @@ func SetLog() {
zapp := zap.New(zap.UseFlagOptions(ZapOpts))
dlog := logf.NewDelegatingLogSink(zapp.GetSink())
ctrl.Log = logr.New(dlog)
SetupLog = ctrl.Log.WithName("setup")
DaemonLog = ctrl.Log.WithName("controllers").WithName("Daemon")
DefLog = ctrl.Log.WithName("controllers").WithName("NetAttachDef")
CIDRLog = ctrl.Log.WithName("controllers").WithName("CIDR")
Expand Down
Loading

0 comments on commit 1dd8ae3

Please sign in to comment.