@@ -37,6 +37,10 @@ import (
37
37
"github.com/gofrs/uuid"
38
38
)
39
39
40
+ const (
41
+ reconfigureWaitTimeout = 5 * time .Minute
42
+ )
43
+
40
44
// cloudbeat configuration.
41
45
type cloudbeat struct {
42
46
ctx context.Context
@@ -122,12 +126,25 @@ func (bt *cloudbeat) Run(b *beat.Beat) error {
122
126
bt .log .Info ("cloudbeat is running! Hit CTRL-C to stop it." )
123
127
124
128
// Configure the beats Manager to start after all the reloadable hooks are initialized
125
- // and shutdown when the function return .
129
+ // and shutdown when the function returns .
126
130
if err := b .Manager .Start (); err != nil {
127
131
return err
128
132
}
129
133
defer b .Manager .Stop ()
130
134
135
+ // Wait for Fleet-side reconfiguration only if cloudbeat is running in Agent-managed mode.
136
+ if b .Manager .Enabled () {
137
+ bt .log .Infof ("Waiting for initial reconfiguration from Fleet server..." )
138
+ update , err := bt .reconfigureWait (reconfigureWaitTimeout )
139
+ if err != nil {
140
+ return err
141
+ }
142
+
143
+ if err := bt .configUpdate (update ); err != nil {
144
+ return fmt .Errorf ("failed to update with initial reconfiguration from Fleet server: %w" , err )
145
+ }
146
+ }
147
+
131
148
if err := bt .data .Run (bt .ctx ); err != nil {
132
149
return err
133
150
}
@@ -154,49 +171,97 @@ func (bt *cloudbeat) Run(b *beat.Beat) error {
154
171
return nil
155
172
156
173
case update := <- bt .configUpdates :
157
- if err := bt .config .Update (update ); err != nil {
158
- bt .log .Errorf ("Could not update cloudbeat config: %v" , err )
159
- break
174
+ if err := bt .configUpdate (update ); err != nil {
175
+ bt .log .Errorf ("Failed to update cloudbeat config: %v" , err )
160
176
}
161
177
162
- policies , err := csppolicies .CISKubernetes ()
163
- if err != nil {
164
- bt .log .Errorf ("Could not load CIS Kubernetes policies: %v" , err )
165
- break
166
- }
178
+ case fetchedResources := <- output :
179
+ cycleId , _ := uuid .NewV4 ()
180
+ cycleStart := time .Now ()
181
+ bt .log .Infof ("Eval cycle %v has started" , cycleId )
182
+
183
+ cycleMetadata := transformer.CycleMetadata {CycleId : cycleId }
184
+ // TODO: send events through a channel and publish them by a configured threshold & time
185
+ events := bt .transformer .ProcessAggregatedResources (fetchedResources , cycleMetadata )
186
+
187
+ bt .log .Infof ("Publishing %d events to index" , len (events ))
188
+ bt .client .PublishAll (events )
189
+
190
+ bt .log .Infof ("Eval cycle %v published %d events after %s" , cycleId , len (events ), time .Since (cycleStart ))
191
+ }
192
+ }
193
+ }
194
+
195
+ // reconfigureWait will wait for and consume incoming reconfuration from the Fleet server, and keep
196
+ // discarding them until the incoming config contains the necessary information to start cloudbeat
197
+ // properly, thereafter returning the valid config.
198
+ func (bt * cloudbeat ) reconfigureWait (timeout time.Duration ) (* common.Config , error ) {
199
+ start := time .Now ()
200
+ timer := time .After (timeout )
201
+
202
+ for {
203
+ select {
204
+ case <- bt .ctx .Done ():
205
+ return nil , fmt .Errorf ("cancelled via context" )
167
206
168
- if len (bt .config .Streams ) == 0 {
169
- bt .log .Infof ("Did not receive any input stream, skipping." )
170
- break
207
+ case <- timer :
208
+ return nil , fmt .Errorf ("timed out waiting for reconfiguration" )
209
+
210
+ case update , ok := <- bt .configUpdates :
211
+ if ! ok {
212
+ return nil , fmt .Errorf ("reconfiguration channel is closed" )
171
213
}
172
214
173
- y , err := bt . config .DataYaml ( )
215
+ c , err := config .New ( update )
174
216
if err != nil {
175
- bt .log .Errorf ("Could not marshal to YAML : %v" , err )
176
- break
217
+ bt .log .Errorf ("Could not parse reconfiguration %v, skipping with error : %v" , update . FlattenedKeys () , err )
218
+ continue
177
219
}
178
220
179
- if err := csppolicies . HostBundleWithDataYaml ( "bundle.tar.gz" , policies , y ); err != nil {
180
- bt .log .Errorf ( "Could not update bundle with dataYaml: %v" , err )
181
- break
221
+ if len ( c . Streams ) == 0 {
222
+ bt .log .Infof ( "No streams received in reconfiguration %v" , update . FlattenedKeys () )
223
+ continue
182
224
}
183
225
184
- bt .log .Infof ("Bundle updated with dataYaml: %s" , y )
226
+ if c .Streams [0 ].DataYaml == nil {
227
+ bt .log .Infof ("data_yaml not present in reconfiguration %v" , update .FlattenedKeys ())
228
+ continue
229
+ }
185
230
186
- case fetchedResources := <- output :
187
- cycleId , _ := uuid .NewV4 ()
188
- bt .log .Infof ("Cycle %v has started" , cycleId )
231
+ bt .log .Infof ("Received valid reconfiguration after waiting for %s" , time .Since (start ))
232
+ return update , nil
233
+ }
234
+ }
235
+ }
189
236
190
- cycleMetadata := transformer.CycleMetadata {CycleId : cycleId }
191
- // TODO: send events through a channel and publish them by a configured threshold & time
192
- events := bt .transformer .ProcessAggregatedResources (fetchedResources , cycleMetadata )
237
+ // configUpdate applies incoming reconfiguration from the Fleet server to the cloudbeat config,
238
+ // and updates the hosted bundle with the new values.
239
+ func (bt * cloudbeat ) configUpdate (update * common.Config ) error {
240
+ if err := bt .config .Update (bt .log , update ); err != nil {
241
+ return err
242
+ }
193
243
194
- bt .log .Infof ("Publishing %d events to index" , len (events ))
195
- bt .client .PublishAll (events )
244
+ policies , err := csppolicies .CISKubernetes ()
245
+ if err != nil {
246
+ return fmt .Errorf ("could not load CIS Kubernetes policies: %w" , err )
247
+ }
196
248
197
- bt .log .Infof ("Cycle %v has ended" , cycleId )
198
- }
249
+ if len (bt .config .Streams ) == 0 {
250
+ bt .log .Infof ("Did not receive any input stream from incoming config, skipping." )
251
+ return nil
199
252
}
253
+
254
+ y , err := bt .config .DataYaml ()
255
+ if err != nil {
256
+ return fmt .Errorf ("could not marshal to YAML: %w" , err )
257
+ }
258
+
259
+ if err := csppolicies .HostBundleWithDataYaml ("bundle.tar.gz" , policies , y ); err != nil {
260
+ return fmt .Errorf ("could not update bundle with dataYaml: %w" , err )
261
+ }
262
+
263
+ bt .log .Infof ("Bundle updated with dataYaml: %s" , y )
264
+ return nil
200
265
}
201
266
202
267
func InitRegistry (log * logp.Logger , c config.Config ) (manager.FetchersRegistry , error ) {
0 commit comments