From 9523f1e411b5886f25deafbf7c2e5d438edb948a Mon Sep 17 00:00:00 2001 From: Markus Lehtonen Date: Fri, 21 Apr 2023 19:15:29 +0300 Subject: [PATCH] nfd-master: fix a crash when processing NodeFeatureRules Fix a a bug where nfd-master with NodeFeature API enabled would crash when NodeFeatureRule objects were processed in the case where no NodeFeature objects existed. This was caused by trying to insert values into a non-initialized NodeFeatureSpec in the code. This patch adds two safety measures to prevent that from happening in the future. First, add a constructor function for the NodeFeatureSpec type, and second, check for uninitialized object in the function inserting new functions. TODO: add unit tests for the API helper functions. --- pkg/apis/nfd/v1alpha1/feature.go | 19 +++++++++++++++++++ pkg/nfd-master/nfd-master.go | 3 ++- 2 files changed, 21 insertions(+), 1 deletion(-) diff --git a/pkg/apis/nfd/v1alpha1/feature.go b/pkg/apis/nfd/v1alpha1/feature.go index 8615cdf64c..6f112d83e5 100644 --- a/pkg/apis/nfd/v1alpha1/feature.go +++ b/pkg/apis/nfd/v1alpha1/feature.go @@ -16,6 +16,15 @@ limitations under the License. package v1alpha1 +// NewNodeFeatureSpec creates a new emprty instance of NodeFeatureSpec type, +// initializing all fields to proper empty values. +func NewNodeFeatureSpec() *NodeFeatureSpec { + return &NodeFeatureSpec{ + Features: *NewFeatures(), + Labels: make(map[string]string), + } +} + // NewFeatures creates a new instance of Features, initializing all feature // types (flags, attributes and instances) to empty values. func NewFeatures() *Features { @@ -56,7 +65,11 @@ func NewInstanceFeature(attrs map[string]string) *InstanceFeature { } // InsertAttributeFeatures inserts new values into a specific feature. +// TODO: add unit tests func (f *Features) InsertAttributeFeatures(domain, feature string, values map[string]string) { + if f.Attributes == nil { + f.Attributes = make(map[string]AttributeFeatureSet) + } key := domain + "." + feature if _, ok := f.Attributes[key]; !ok { f.Attributes[key] = NewAttributeFeatures(values) @@ -70,6 +83,7 @@ func (f *Features) InsertAttributeFeatures(domain, feature string, values map[st // Exists returns a non-empty string if a feature exists. The return value is // the type of the feautre, i.e. "flag", "attribute" or "instance". +// TODO: add unit tests func (f *Features) Exists(name string) string { if _, ok := f.Flags[name]; ok { return "flag" @@ -85,6 +99,7 @@ func (f *Features) Exists(name string) string { // MergeInto merges two FeatureSpecs into one. Data in the input object takes // precedence (overwrite) over data of the existing object we're merging into. +// TODO: add unit tests func (in *NodeFeatureSpec) MergeInto(out *NodeFeatureSpec) { in.Features.MergeInto(&out.Features) if in.Labels != nil { @@ -100,6 +115,7 @@ func (in *NodeFeatureSpec) MergeInto(out *NodeFeatureSpec) { // MergeInto merges two sets of features into one. Features from the input set // take precedence (overwrite) features from the existing features of the set // we're merging into. +// TODO: add unit tests func (in *Features) MergeInto(out *Features) { if in.Flags != nil { if out.Flags == nil { @@ -134,6 +150,7 @@ func (in *Features) MergeInto(out *Features) { } // MergeInto merges two sets of flag featues. +// TODO: add unit tests func (in *FlagFeatureSet) MergeInto(out *FlagFeatureSet) { if in.Elements != nil { if out.Elements == nil { @@ -146,6 +163,7 @@ func (in *FlagFeatureSet) MergeInto(out *FlagFeatureSet) { } // MergeInto merges two sets of attribute featues. +// TODO: add unit tests func (in *AttributeFeatureSet) MergeInto(out *AttributeFeatureSet) { if in.Elements != nil { if out.Elements == nil { @@ -158,6 +176,7 @@ func (in *AttributeFeatureSet) MergeInto(out *AttributeFeatureSet) { } // MergeInto merges two sets of instance featues. +// TODO: add unit tests func (in *InstanceFeatureSet) MergeInto(out *InstanceFeatureSet) { if in.Elements != nil { if out.Elements == nil { diff --git a/pkg/nfd-master/nfd-master.go b/pkg/nfd-master/nfd-master.go index 6e8bed6df7..1fb04ba7b5 100644 --- a/pkg/nfd-master/nfd-master.go +++ b/pkg/nfd-master/nfd-master.go @@ -649,7 +649,8 @@ func (m *nfdMaster) nfdAPIUpdateOneNode(nodeName string) error { klog.V(1).Infof("processing node %q, initiated by NodeFeature API", nodeName) - features := &nfdv1alpha1.NodeFeatureSpec{} + features := nfdv1alpha1.NewNodeFeatureSpec() + annotations := Annotations{} if len(objs) > 0 {