Skip to content

Commit

Permalink
fix: do not try to notify if required data are missing
Browse files Browse the repository at this point in the history
Adds a check which controls if notification will happen or not. The
check validates presence of the most required parts and creates a basis
for adding more.

Now it checks for:
- having at least one sample to send
- HostInfo present and containing HostID and ExternalOrganization

Signed-off-by: Petr Vobornik <[email protected]>
  • Loading branch information
pvoborni committed Oct 20, 2023
1 parent 0de885a commit f18b36e
Show file tree
Hide file tree
Showing 4 changed files with 180 additions and 9 deletions.
10 changes: 7 additions & 3 deletions daemon/daemon.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ type Daemon struct {
metricsLog *notify.MetricsLog
certWatcher hostinfo.CertWatcher
notifier notify.Notifier
notifyPolicy notify.NotifyPolicy
stopCh chan os.Signal
started bool
}
Expand All @@ -30,6 +31,7 @@ func NewDaemon(config *config.Config) (*Daemon, error) {
config: config,
notifier: notify.NewPrometheusNotifier(config),
hostInfoProvider: &hostinfo.SubManInfoProvider{},
notifyPolicy: &notify.GeneralNotifyPolicy{},
}
d.certWatcher, err = hostinfo.NewINotifyCertWatcher(d.config.HostCertPath)
if err != nil {
Expand Down Expand Up @@ -208,11 +210,13 @@ func (d *Daemon) notify() error {
if d.config.MetricsMaxAge > 0 {
samples = notify.FilterSamplesByAge(samples, d.config.MetricsMaxAge)
}
count := len(samples)
if count == 0 {
logger.Debugln("No samples to send")
err = d.notifyPolicy.ShouldNotify(samples, d.hostInfo)
if err != nil {
logger.Errorf("Cannot notify: %s\n", err.Error())
return nil
}

count := len(samples)
logger.Debugf("Sending %d sample(s)...\n", count)
err = d.notifier.Notify(samples, d.hostInfo)
if err != nil {
Expand Down
42 changes: 36 additions & 6 deletions daemon/daemon_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -152,13 +152,15 @@ func TestNotify(t *testing.T) {
daemon, notifier, metricsLog, hiProvider := createDaemon(t)
daemon.config.MetricsMaxAge = 10 * time.Second
daemon.hostInfo, _ = hiProvider.Load()
notifyPolicy := daemon.notifyPolicy.(*mockNotifyPolicy)

// Test that notifier is called when there are some samples
metricsLog.WriteSampleNow(1)
notifier.ExpectSuccess()
err := daemon.notify()
checkError(t, err, "failed to notify")
notifier.CheckWasCalled(t)
notifyPolicy.CheckWasCalled(t)

// Test that notifier was called with the sample and hostinfo
if len(notifier.calledWith.samples) != 1 {
Expand Down Expand Up @@ -335,14 +337,16 @@ func createDaemon(t *testing.T) (*Daemon, *mockNotifier, *notify.MetricsLog, *mo
notifier := &mockNotifier{}
notifier.ExpectSuccess()
daemon.notifier = notifier
daemon.notifyPolicy = NewMockNotifyPolicy(false)
daemon.certWatcher = &mockCertWatcher{make(chan hostinfo.CertEvent)}
hiProvider := newMockHostInfoProvider(&hostinfo.HostInfo{
CpuCount: 2,
HostId: "testhost-id",
SocketCount: "1",
Product: "testproduct",
Support: "testsupport",
Usage: "testusage",
CpuCount: 2,
HostId: "testhost-id",
ExternalOrganization: "testorg",
SocketCount: "1",
Product: "testproduct",
Support: "testsupport",
Usage: "testusage",
Billing: hostinfo.BillingInfo{
Model: "testmodel",
Marketplace: "testmarketplace",
Expand Down Expand Up @@ -426,6 +430,32 @@ func (n *mockNotifier) WaitForCall(t *testing.T, timeout time.Duration) {
}
}

// Mock NotifyPolicy
type mockNotifyPolicy struct {
Called uint
Fake bool
}

func (m *mockNotifyPolicy) ShouldNotify(samples []prompb.Sample, hostinfo *hostinfo.HostInfo) error {
m.Called++
if !m.Fake {
policy := &notify.GeneralNotifyPolicy{}
return policy.ShouldNotify(samples, hostinfo)
}
return nil
}

func (m *mockNotifyPolicy) CheckWasCalled(t *testing.T) {
t.Helper()
if m.Called == 0 {
t.Fatalf("expected notify policy to be called")
}
}

func NewMockNotifyPolicy(fake bool) *mockNotifyPolicy {
return &mockNotifyPolicy{0, fake}
}

// Mock HostInfo provider

type mockHostInfoProvider struct {
Expand Down
37 changes: 37 additions & 0 deletions notify/policy.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
package notify

import (
"fmt"

"github.com/prometheus/prometheus/prompb"

"github.com/RedHatInsights/host-metering/hostinfo"
)

type NotifyPolicy interface {
ShouldNotify([]prompb.Sample, *hostinfo.HostInfo) error
}

type GeneralNotifyPolicy struct{}

func (p *GeneralNotifyPolicy) ShouldNotify(samples []prompb.Sample, hostinfo *hostinfo.HostInfo) error {

count := len(samples)
if count == 0 {
return fmt.Errorf("no samples to send")
}

if hostinfo == nil {
return fmt.Errorf("missing internal HostInfo")
}

if hostinfo.HostId == "" {
return fmt.Errorf("missing HostId")
}

if hostinfo.ExternalOrganization == "" {
return fmt.Errorf("missing ExternalOrganization")
}

return nil
}
100 changes: 100 additions & 0 deletions notify/policy_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,100 @@
package notify

import (
"strings"
"testing"

"github.com/RedHatInsights/host-metering/hostinfo"
"github.com/prometheus/prometheus/prompb"
)

type ShouldNotifyTestCase struct {
name string
samples []prompb.Sample
hostInfo *hostinfo.HostInfo
expected string
}

func TestGeneralNotifyPolicy(t *testing.T) {

p := &GeneralNotifyPolicy{}
correctSamples := []prompb.Sample{{}}
correctHostInfo := fullyDefinedMockHostInfo()

emptyHostIdHI := fullyDefinedMockHostInfo()
emptyHostIdHI.HostId = ""

emptyExternalOrganizationHI := fullyDefinedMockHostInfo()
emptyExternalOrganizationHI.ExternalOrganization = ""

testCases := []ShouldNotifyTestCase{
{
name: "No samples",
samples: []prompb.Sample{},
hostInfo: correctHostInfo,
expected: "no samples to send",
},
{
name: "Nil hostInfo",
samples: correctSamples,
hostInfo: nil,
expected: "missing internal HostInfo",
},
{
name: "Empty HostId",
samples: correctSamples,
hostInfo: emptyHostIdHI,
expected: "missing HostId",
},
{
name: "Empty ExternalOrganization",
samples: correctSamples,
hostInfo: emptyExternalOrganizationHI,
expected: "missing ExternalOrganization",
},
}

t.Run("Test ShouldNotify", func(t *testing.T) {

for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {
// when
err := p.ShouldNotify(tc.samples, tc.hostInfo)

// then
expectErrorContains(t, err, tc.expected)
})
}
})
}

func fullyDefinedMockHostInfo() *hostinfo.HostInfo {
return &hostinfo.HostInfo{
HostId: "hostid",
ExternalOrganization: "externalorganization",
SocketCount: "socketcount",
Product: "product",
Support: "support",
Usage: "usage",
Billing: fullyDefinedMockBillingInfo(),
}
}

func fullyDefinedMockBillingInfo() hostinfo.BillingInfo {
return hostinfo.BillingInfo{
Model: "model",
Marketplace: "marketplace",
MarketplaceAccount: "marketplaceaccount",
MarketplaceInstanceId: "marketplaceinstanceid",
}
}

func expectErrorContains(t *testing.T, err error, expected string) {
t.Helper()
if err == nil {
t.Fatalf("expected error, got nil")
}
if !strings.Contains(err.Error(), expected) {
t.Fatalf("expected error to contain '%s', got '%s'", expected, err.Error())
}
}

0 comments on commit f18b36e

Please sign in to comment.