-
Notifications
You must be signed in to change notification settings - Fork 4.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Additional Heartbeat Reload Tests (#8228)
* Adds a mocked test of the Monitor type * Minimizes the required mockable area by introducing a PipelineConnector type that is easier to mock than a Pipeline. * Introduces a number of useful mocks for future tests potentially.
- Loading branch information
Showing
6 changed files
with
218 additions
and
13 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,124 @@ | ||
// Licensed to Elasticsearch B.V. under one or more contributor | ||
// license agreements. See the NOTICE file distributed with | ||
// this work for additional information regarding copyright | ||
// ownership. Elasticsearch B.V. licenses this file to you under | ||
// the Apache License, Version 2.0 (the "License"); you may | ||
// not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, | ||
// software distributed under the License is distributed on an | ||
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | ||
// KIND, either express or implied. See the License for the | ||
// specific language governing permissions and limitations | ||
// under the License. | ||
|
||
package monitors | ||
|
||
import ( | ||
"fmt" | ||
"sync" | ||
"testing" | ||
|
||
"github.com/stretchr/testify/require" | ||
|
||
"github.com/elastic/beats/libbeat/beat" | ||
"github.com/elastic/beats/libbeat/common" | ||
) | ||
|
||
type MockBeatClient struct { | ||
publishes []beat.Event | ||
closed bool | ||
mtx sync.Mutex | ||
} | ||
|
||
func (c *MockBeatClient) Publish(e beat.Event) { | ||
c.PublishAll([]beat.Event{e}) | ||
} | ||
|
||
func (c *MockBeatClient) PublishAll(events []beat.Event) { | ||
c.mtx.Lock() | ||
defer c.mtx.Unlock() | ||
|
||
for _, e := range events { | ||
c.publishes = append(c.publishes, e) | ||
} | ||
} | ||
|
||
func (c *MockBeatClient) Close() error { | ||
c.mtx.Lock() | ||
defer c.mtx.Unlock() | ||
|
||
if c.closed { | ||
return fmt.Errorf("mock client already closed") | ||
} | ||
|
||
c.closed = true | ||
return nil | ||
} | ||
|
||
func (c *MockBeatClient) Publishes() []beat.Event { | ||
c.mtx.Lock() | ||
defer c.mtx.Unlock() | ||
|
||
dst := make([]beat.Event, len(c.publishes)) | ||
copy(dst, c.publishes) | ||
return dst | ||
} | ||
|
||
type MockPipelineConnector struct { | ||
clients []*MockBeatClient | ||
mtx sync.Mutex | ||
} | ||
|
||
func (pc *MockPipelineConnector) Connect() (beat.Client, error) { | ||
return pc.ConnectWith(beat.ClientConfig{}) | ||
} | ||
|
||
func (pc *MockPipelineConnector) ConnectWith(beat.ClientConfig) (beat.Client, error) { | ||
pc.mtx.Lock() | ||
defer pc.mtx.Unlock() | ||
|
||
c := &MockBeatClient{} | ||
|
||
pc.clients = append(pc.clients, c) | ||
|
||
return c, nil | ||
} | ||
|
||
func createMockJob(name string, cfg *common.Config) ([]Job, error) { | ||
j := MakeSimpleJob(JobSettings{}, func() (common.MapStr, error) { | ||
return common.MapStr{ | ||
"foo": "bar", | ||
}, nil | ||
}) | ||
|
||
return []Job{j}, nil | ||
} | ||
|
||
func mockPluginBuilder() pluginBuilder { | ||
return pluginBuilder{"test", ActiveMonitor, func(s string, config *common.Config) ([]Job, error) { | ||
c := common.Config{} | ||
j, err := createMockJob("test", &c) | ||
return j, err | ||
}} | ||
} | ||
|
||
func mockPluginsReg() *pluginsReg { | ||
reg := newPluginsReg() | ||
reg.add(mockPluginBuilder()) | ||
return reg | ||
} | ||
|
||
func mockPluginConf(t *testing.T, schedule string, url string) *common.Config { | ||
conf, err := common.NewConfigFrom(map[string]interface{}{ | ||
"type": "test", | ||
"urls": []string{url}, | ||
"schedule": schedule, | ||
}) | ||
require.NoError(t, err) | ||
|
||
return conf | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,69 @@ | ||
// Licensed to Elasticsearch B.V. under one or more contributor | ||
// license agreements. See the NOTICE file distributed with | ||
// this work for additional information regarding copyright | ||
// ownership. Elasticsearch B.V. licenses this file to you under | ||
// the Apache License, Version 2.0 (the "License"); you may | ||
// not use this file except in compliance with the License. | ||
// You may obtain a copy of the License at | ||
// | ||
// http://www.apache.org/licenses/LICENSE-2.0 | ||
// | ||
// Unless required by applicable law or agreed to in writing, | ||
// software distributed under the License is distributed on an | ||
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | ||
// KIND, either express or implied. See the License for the | ||
// specific language governing permissions and limitations | ||
// under the License. | ||
|
||
package monitors | ||
|
||
import ( | ||
"testing" | ||
"time" | ||
|
||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
|
||
"github.com/elastic/beats/heartbeat/scheduler" | ||
) | ||
|
||
func TestMonitor(t *testing.T) { | ||
serverMonConf := mockPluginConf(t, "@every 1ms", "http://example.net") | ||
reg := mockPluginsReg() | ||
pipelineConnector := &MockPipelineConnector{} | ||
|
||
sched := scheduler.New(1) | ||
err := sched.Start() | ||
require.NoError(t, err) | ||
defer sched.Stop() | ||
|
||
mon, err := newMonitor(serverMonConf, reg, pipelineConnector, sched, false) | ||
require.NoError(t, err) | ||
|
||
mon.Start() | ||
defer mon.Stop() | ||
|
||
require.Equal(t, 1, len(pipelineConnector.clients)) | ||
pcClient := pipelineConnector.clients[0] | ||
|
||
timeout := time.Second | ||
start := time.Now() | ||
success := false | ||
for time.Since(start) < timeout && !success { | ||
count := len(pcClient.Publishes()) | ||
if count >= 1 { | ||
success = true | ||
} else { | ||
// Let's yield this goroutine so we don't spin | ||
// This could (possibly?) lock on a single core system otherwise | ||
time.Sleep(time.Microsecond) | ||
} | ||
} | ||
|
||
if !success { | ||
t.Fatalf("No publishes detected!") | ||
} | ||
|
||
mon.Stop() | ||
assert.Equal(t, true, pcClient.closed) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters