5652bde447
* Use secrets service in pluginproxy
* Use secrets service in pluginxontext
* Use secrets service in pluginsettings
* Use secrets service in provisioning
* Use secrets service in authinfoservice
* Use secrets service in api
* Use secrets service in sqlstore
* Use secrets service in dashboardshapshots
* Use secrets service in tsdb
* Use secrets service in datasources
* Use secrets service in alerting
* Use secrets service in ngalert
* Break cyclic dependancy
* Refactor service
* Break cyclic dependancy
* Add FakeSecretsStore
* Setup Secrets Service in sqlstore
* Fix
* Continue secrets service refactoring
* Fix cyclic dependancy in sqlstore tests
* Fix secrets service references
* Fix linter errors
* Add fake secrets service for tests
* Refactor SetupTestSecretsService
* Update setting up secret service in tests
* Fix missing secrets service in multiorg_alertmanager_test
* Use fake db in tests and sort imports
* Use fake db in datasources tests
* Fix more tests
* Fix linter issues
* Attempt to fix plugin proxy tests
* Pass secrets service to getPluginProxiedRequest in pluginproxy tests
* Fix pluginproxy tests
* Revert using secrets service in alerting and provisioning
* Update decryptFn in alerting migration
* Rename defaultProvider to currentProvider
* Use fake secrets service in alert channels tests
* Refactor secrets service test helper
* Update setting up secrets service in tests
* Revert alerting changes in api
* Add comments
* Remove secrets service from background services
* Convert global encryption functions into vars
* Revert "Convert global encryption functions into vars"
This reverts commit 498eb19859.
* Add feature toggle for envelope encryption
* Rename toggle
Co-authored-by: Emil Tullstedt <emil.tullstedt@grafana.com>
Co-authored-by: Joan López de la Franca Beltran <joanjan14@gmail.com>
198 lines
5.5 KiB
Go
198 lines
5.5 KiB
Go
package channels
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"mime/multipart"
|
|
"net/url"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/grafana/grafana/pkg/bus"
|
|
"github.com/grafana/grafana/pkg/components/simplejson"
|
|
"github.com/grafana/grafana/pkg/models"
|
|
"github.com/grafana/grafana/pkg/services/secrets/fakes"
|
|
secretsManager "github.com/grafana/grafana/pkg/services/secrets/manager"
|
|
|
|
"github.com/prometheus/alertmanager/notify"
|
|
"github.com/prometheus/alertmanager/types"
|
|
"github.com/prometheus/common/model"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestPushoverNotifier(t *testing.T) {
|
|
tmpl := templateForTests(t)
|
|
|
|
externalURL, err := url.Parse("http://localhost")
|
|
require.NoError(t, err)
|
|
tmpl.ExternalURL = externalURL
|
|
|
|
cases := []struct {
|
|
name string
|
|
settings string
|
|
alerts []*types.Alert
|
|
expMsg map[string]string
|
|
expInitError string
|
|
expMsgError error
|
|
}{
|
|
{
|
|
name: "Correct config with one alert",
|
|
settings: `{
|
|
"userKey": "<userKey>",
|
|
"apiToken": "<apiToken>"
|
|
}`,
|
|
alerts: []*types.Alert{
|
|
{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{"__alert_rule_uid__": "rule uid", "alertname": "alert1", "lbl1": "val1"},
|
|
Annotations: model.LabelSet{"ann1": "annv1", "__dashboardUid__": "abcd", "__panelId__": "efgh"},
|
|
},
|
|
},
|
|
},
|
|
expMsg: map[string]string{
|
|
"user": "<userKey>",
|
|
"token": "<apiToken>",
|
|
"priority": "0",
|
|
"sound": "",
|
|
"title": "[FIRING:1] (val1)",
|
|
"url": "http://localhost/alerting/list",
|
|
"url_title": "Show alert rule",
|
|
"message": "**Firing**\n\nLabels:\n - alertname = alert1\n - lbl1 = val1\nAnnotations:\n - ann1 = annv1\nSilence: http://localhost/alerting/silence/new?alertmanager=grafana&matchers=alertname%3Dalert1%2Clbl1%3Dval1\nDashboard: http://localhost/d/abcd\nPanel: http://localhost/d/abcd?viewPanel=efgh\n",
|
|
"html": "1",
|
|
},
|
|
expMsgError: nil,
|
|
},
|
|
{
|
|
name: "Custom config with multiple alerts",
|
|
settings: `{
|
|
"userKey": "<userKey>",
|
|
"apiToken": "<apiToken>",
|
|
"device": "device",
|
|
"priority": "2",
|
|
"okpriority": "0",
|
|
"retry": "30",
|
|
"expire": "86400",
|
|
"sound": "echo",
|
|
"oksound": "magic",
|
|
"message": "{{ len .Alerts.Firing }} alerts are firing, {{ len .Alerts.Resolved }} are resolved"
|
|
}`,
|
|
alerts: []*types.Alert{
|
|
{
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{"__alert_rule_uid__": "rule uid", "alertname": "alert1", "lbl1": "val1"},
|
|
Annotations: model.LabelSet{"ann1": "annv1"},
|
|
},
|
|
}, {
|
|
Alert: model.Alert{
|
|
Labels: model.LabelSet{"alertname": "alert1", "lbl1": "val2"},
|
|
Annotations: model.LabelSet{"ann1": "annv2"},
|
|
},
|
|
},
|
|
},
|
|
expMsg: map[string]string{
|
|
"user": "<userKey>",
|
|
"token": "<apiToken>",
|
|
"priority": "2",
|
|
"sound": "echo",
|
|
"title": "[FIRING:2] ",
|
|
"url": "http://localhost/alerting/list",
|
|
"url_title": "Show alert rule",
|
|
"message": "2 alerts are firing, 0 are resolved",
|
|
"html": "1",
|
|
"retry": "30",
|
|
"expire": "86400",
|
|
"device": "device",
|
|
},
|
|
expMsgError: nil,
|
|
},
|
|
{
|
|
name: "Missing user key",
|
|
settings: `{
|
|
"apiToken": "<apiToken>"
|
|
}`,
|
|
expInitError: `failed to validate receiver "pushover_testing" of type "pushover": user key not found`,
|
|
}, {
|
|
name: "Missing api key",
|
|
settings: `{
|
|
"userKey": "<userKey>"
|
|
}`,
|
|
expInitError: `failed to validate receiver "pushover_testing" of type "pushover": API token not found`,
|
|
},
|
|
}
|
|
|
|
for _, c := range cases {
|
|
origGetBoundary := GetBoundary
|
|
boundary := "abcd"
|
|
GetBoundary = func() string {
|
|
return boundary
|
|
}
|
|
t.Cleanup(func() {
|
|
GetBoundary = origGetBoundary
|
|
})
|
|
|
|
t.Run(c.name, func(t *testing.T) {
|
|
settingsJSON, err := simplejson.NewJson([]byte(c.settings))
|
|
require.NoError(t, err)
|
|
|
|
m := &NotificationChannelConfig{
|
|
Name: "pushover_testing",
|
|
Type: "pushover",
|
|
Settings: settingsJSON,
|
|
}
|
|
|
|
secretsService := secretsManager.SetupTestService(t, fakes.NewFakeSecretsStore())
|
|
decryptFn := secretsService.GetDecryptedValue
|
|
pn, err := NewPushoverNotifier(m, tmpl, decryptFn)
|
|
if c.expInitError != "" {
|
|
require.Error(t, err)
|
|
require.Equal(t, c.expInitError, err.Error())
|
|
return
|
|
}
|
|
require.NoError(t, err)
|
|
|
|
body := ""
|
|
bus.AddHandlerCtx("test", func(ctx context.Context, webhook *models.SendWebhookSync) error {
|
|
body = webhook.Body
|
|
return nil
|
|
})
|
|
|
|
ctx := notify.WithGroupKey(context.Background(), "alertname")
|
|
ctx = notify.WithGroupLabels(ctx, model.LabelSet{"alertname": ""})
|
|
ok, err := pn.Notify(ctx, c.alerts...)
|
|
if c.expMsgError != nil {
|
|
require.Error(t, err)
|
|
require.False(t, ok)
|
|
require.Equal(t, c.expMsgError.Error(), err.Error())
|
|
return
|
|
}
|
|
require.NoError(t, err)
|
|
require.True(t, ok)
|
|
|
|
bodyReader := multipart.NewReader(strings.NewReader(body), boundary)
|
|
for {
|
|
part, err := bodyReader.NextPart()
|
|
if part == nil || errors.Is(err, io.EOF) {
|
|
assert.Empty(t, c.expMsg, fmt.Sprintf("expected fields %v", c.expMsg))
|
|
break
|
|
}
|
|
formField := part.FormName()
|
|
expected, ok := c.expMsg[formField]
|
|
assert.True(t, ok, fmt.Sprintf("unexpected field %s", formField))
|
|
actual := []byte("")
|
|
if expected != "" {
|
|
buf := new(bytes.Buffer)
|
|
_, err := buf.ReadFrom(part)
|
|
require.NoError(t, err)
|
|
actual = buf.Bytes()
|
|
}
|
|
assert.Equal(t, expected, string(actual))
|
|
delete(c.expMsg, formField)
|
|
}
|
|
})
|
|
}
|
|
}
|