Skip to content

Commit 22b34ef

Browse files
authored
Fix bug in Config.Update (#149)
Incoming config updates with slices smaller than the current slice didn't resize the current slice, resulting in undesirable values in the tail of the slice.
1 parent e061084 commit 22b34ef

File tree

2 files changed

+220
-8
lines changed

2 files changed

+220
-8
lines changed

config/config.go

+22
Original file line numberDiff line numberDiff line change
@@ -63,11 +63,33 @@ func New(cfg *common.Config) (Config, error) {
6363
return c, nil
6464
}
6565

66+
// Update replaces values of those keys in the current config which are
67+
// present in the incoming config.
68+
//
69+
// NOTE(yashtewari): This will be removed with the planned update to restart the
70+
// beat with the new config.
6671
func (c *Config) Update(cfg *common.Config) error {
6772
if err := cfg.Unpack(&c); err != nil {
6873
return err
6974
}
7075

76+
// Check if the incoming config has streams.
77+
m := make(map[string]interface{})
78+
if err := cfg.Unpack(&m); err != nil {
79+
return err
80+
}
81+
82+
if _, ok := m["streams"]; !ok {
83+
return nil
84+
}
85+
86+
uc, err := New(cfg)
87+
if err != nil {
88+
return err
89+
}
90+
91+
c.Streams = uc.Streams
92+
7193
return nil
7294
}
7395

config/config_test.go

+198-8
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ package config
2323
import (
2424
"strings"
2525
"testing"
26+
"time"
2627

2728
"github.com/elastic/beats/v7/libbeat/common"
2829
"github.com/stretchr/testify/suite"
@@ -41,8 +42,8 @@ func (s *ConfigTestSuite) SetupTest() {
4142

4243
func (s *ConfigTestSuite) TestNew() {
4344
var tests = []struct {
44-
config string
45-
expectedPatterns []string
45+
config string
46+
expected []string
4647
}{
4748
{
4849
`
@@ -54,8 +55,9 @@ func (s *ConfigTestSuite) TestNew() {
5455
- b
5556
- c
5657
- d
58+
- e
5759
`,
58-
[]string{"a", "b", "c", "d"},
60+
[]string{"a", "b", "c", "d", "e"},
5961
},
6062
}
6163

@@ -66,14 +68,202 @@ func (s *ConfigTestSuite) TestNew() {
6668
c, err := New(cfg)
6769
s.NoError(err)
6870

69-
s.Equal(test.expectedPatterns, c.Streams[0].DataYaml.ActivatedRules.CISK8S)
71+
s.Equal(test.expected, c.Streams[0].DataYaml.ActivatedRules.CISK8S)
72+
}
73+
}
74+
75+
func (s *ConfigTestSuite) TestConfigUpdate() {
76+
config := `
77+
streams:
78+
- data_yaml:
79+
activated_rules:
80+
cis_k8s:
81+
- a
82+
- b
83+
- c
84+
- d
85+
- e`
86+
87+
var tests = []struct {
88+
update string
89+
expected []string
90+
}{
91+
{
92+
`
93+
streams:
94+
- data_yaml:
95+
activated_rules:
96+
cis_k8s:
97+
- a
98+
- b
99+
- c
100+
- d
101+
`,
102+
[]string{"a", "b", "c", "d"},
103+
},
104+
{
105+
`
106+
streams:
107+
- data_yaml:
108+
activated_rules:
109+
cis_k8s:
110+
- b
111+
- c
112+
- d
113+
- e
114+
- f
115+
`,
116+
[]string{"b", "c", "d", "e", "f"},
117+
},
118+
{
119+
`
120+
streams:
121+
- data_yaml:
122+
activated_rules:
123+
cis_k8s: []
124+
`,
125+
[]string{},
126+
},
127+
{
128+
`
129+
streams:
130+
- data_yaml:
131+
activated_rules:
132+
cis_k8s:
133+
- a
134+
- b
135+
- c
136+
- d
137+
- e
138+
`,
139+
[]string{"a", "b", "c", "d", "e"},
140+
},
141+
// We're not currently aware of a scenario where we receive
142+
// multiple input streams, but still to make sure that it doesn't
143+
// break for this scenario.
144+
{
145+
`
146+
streams:
147+
- data_yaml:
148+
activated_rules:
149+
cis_k8s:
150+
- x
151+
- "y" # Just YAML 1.1 things
152+
- z
153+
- data_yaml:
154+
activated_rules:
155+
- f
156+
- g
157+
- h
158+
- i
159+
- j
160+
- k
161+
`,
162+
[]string{"x", "y", "z"},
163+
},
164+
}
165+
166+
cfg, err := common.NewConfigFrom(config)
167+
s.NoError(err)
168+
169+
c, err := New(cfg)
170+
s.NoError(err)
171+
172+
for _, test := range tests {
173+
cfg, err := common.NewConfigFrom(test.update)
174+
s.NoError(err)
175+
176+
err = c.Update(cfg)
177+
s.NoError(err)
178+
179+
s.Equal(test.expected, c.Streams[0].DataYaml.ActivatedRules.CISK8S)
180+
}
181+
}
182+
183+
// TestConfigUpdateIsolated tests whether updates made to a config from
184+
// are isolated; only those parts of the config specified in the incoming
185+
// config should get updated.
186+
func (s *ConfigTestSuite) TestConfigUpdateIsolated() {
187+
config := `
188+
period: 10s
189+
kube_config: some_path
190+
streams:
191+
- data_yaml:
192+
activated_rules:
193+
cis_k8s:
194+
- a
195+
- b
196+
- c
197+
- d
198+
- e`
199+
200+
var tests = []struct {
201+
update string
202+
expectedPeriod time.Duration
203+
expectedKubeConfig string
204+
expectedCISK8S []string
205+
}{
206+
{
207+
update: `
208+
streams:
209+
- data_yaml:
210+
activated_rules:
211+
cis_k8s:
212+
- a
213+
- b
214+
- c
215+
- d`,
216+
expectedPeriod: 10 * time.Second,
217+
expectedKubeConfig: "some_path",
218+
expectedCISK8S: []string{"a", "b", "c", "d"},
219+
},
220+
{
221+
update: `period: 4h`,
222+
expectedPeriod: 4 * time.Hour,
223+
expectedKubeConfig: "some_path",
224+
expectedCISK8S: []string{"a", "b", "c", "d"},
225+
},
226+
{
227+
update: `
228+
kube_config: some_other_path
229+
streams:
230+
- data_yaml:
231+
activated_rules:
232+
cis_k8s:
233+
- a
234+
- b
235+
- c
236+
- d
237+
- e`,
238+
expectedPeriod: 4 * time.Hour,
239+
expectedKubeConfig: "some_other_path",
240+
expectedCISK8S: []string{"a", "b", "c", "d", "e"},
241+
},
242+
}
243+
244+
cfg, err := common.NewConfigFrom(config)
245+
s.NoError(err)
246+
247+
c, err := New(cfg)
248+
s.NoError(err)
249+
250+
for _, test := range tests {
251+
cfg, err := common.NewConfigFrom(test.update)
252+
s.NoError(err)
253+
254+
err = c.Update(cfg)
255+
s.NoError(err)
256+
257+
s.Equal(test.expectedPeriod, c.Period)
258+
s.Equal(test.expectedKubeConfig, c.KubeConfig)
259+
s.Equal(test.expectedCISK8S, c.Streams[0].DataYaml.ActivatedRules.CISK8S)
70260
}
71261
}
72262

73-
func (s *ConfigTestSuite) TestDataYaml() {
263+
func (s *ConfigTestSuite) TestConfigDataYaml() {
74264
var tests = []struct {
75-
config string
76-
expectedYaml string
265+
config string
266+
expected string
77267
}{
78268
{
79269
`
@@ -107,6 +297,6 @@ activated_rules:
107297
dy, err := c.DataYaml()
108298
s.NoError(err)
109299

110-
s.Equal(strings.TrimSpace(test.expectedYaml), strings.TrimSpace(dy))
300+
s.Equal(strings.TrimSpace(test.expected), strings.TrimSpace(dy))
111301
}
112302
}

0 commit comments

Comments
 (0)