summaryrefslogtreecommitdiffstats
path: root/tests/internal
diff options
context:
space:
mode:
Diffstat (limited to 'tests/internal')
-rw-r--r--tests/internal/utils/database.go28
-rw-r--r--tests/internal/utils/redis.go37
-rw-r--r--tests/internal/utils/slice.go37
-rw-r--r--tests/internal/utils/slice_test.go31
-rw-r--r--tests/internal/value/notification_states.go37
-rw-r--r--tests/internal/value/notification_types.go40
-rw-r--r--tests/internal/value/value.go103
7 files changed, 313 insertions, 0 deletions
diff --git a/tests/internal/utils/database.go b/tests/internal/utils/database.go
new file mode 100644
index 0000000..c32566b
--- /dev/null
+++ b/tests/internal/utils/database.go
@@ -0,0 +1,28 @@
+package utils
+
+import (
+ "fmt"
+ "github.com/icinga/icinga-testing"
+ "github.com/icinga/icinga-testing/services"
+ "os"
+ "strings"
+ "testing"
+)
+
+func GetDatabase(it *icingatesting.IT, t testing.TB) services.RelationalDatabase {
+ k := "ICINGADB_TESTS_DATABASE_TYPE"
+ v := strings.ToLower(os.Getenv(k))
+
+ var rdb services.RelationalDatabase
+
+ switch v {
+ case "mysql":
+ rdb = it.MysqlDatabaseT(t)
+ case "pgsql":
+ rdb = it.PostgresqlDatabaseT(t)
+ default:
+ panic(fmt.Sprintf(`unknown database in %s environment variable: %q (must be "mysql" or "pgsql")`, k, v))
+ }
+
+ return rdb
+}
diff --git a/tests/internal/utils/redis.go b/tests/internal/utils/redis.go
new file mode 100644
index 0000000..da3615d
--- /dev/null
+++ b/tests/internal/utils/redis.go
@@ -0,0 +1,37 @@
+package utils
+
+import (
+ "context"
+ "encoding/hex"
+ "encoding/json"
+ "github.com/go-redis/redis/v8"
+ "github.com/icinga/icinga-testing/services"
+ "github.com/stretchr/testify/require"
+ "testing"
+ "time"
+)
+
+func GetEnvironmentIdFromRedis(t *testing.T, r services.RedisServer) []byte {
+ conn := r.Open()
+ defer conn.Close()
+
+ heartbeat, err := conn.XRead(context.Background(), &redis.XReadArgs{
+ Streams: []string{"icinga:stats", "0"},
+ Count: 1,
+ Block: 10 * time.Second,
+ }).Result()
+ require.NoError(t, err, "reading from icinga:stats failed")
+ require.NotEmpty(t, heartbeat, "response contains no streams")
+ require.NotEmpty(t, heartbeat[0].Messages, "response contains no messages")
+ require.Contains(t, heartbeat[0].Messages[0].Values, "icingadb_environment",
+ "icinga:stats message misses icingadb_environment")
+
+ var envIdHex string
+ err = json.Unmarshal([]byte(heartbeat[0].Messages[0].Values["icingadb_environment"].(string)), &envIdHex)
+ require.NoError(t, err, "cannot parse environment ID as a JSON string")
+
+ envId, err := hex.DecodeString(envIdHex)
+ require.NoError(t, err, "environment ID is not a hex string")
+
+ return envId
+}
diff --git a/tests/internal/utils/slice.go b/tests/internal/utils/slice.go
new file mode 100644
index 0000000..85821f5
--- /dev/null
+++ b/tests/internal/utils/slice.go
@@ -0,0 +1,37 @@
+package utils
+
+import (
+ "fmt"
+ "reflect"
+)
+
+// AnySliceToInterfaceSlice takes a slice of type []T for any T and returns a slice of type []interface{} containing
+// the same elements, somewhat like casting []T to []interface{}.
+func AnySliceToInterfaceSlice(in interface{}) []interface{} {
+ v := reflect.ValueOf(in)
+ if v.Kind() != reflect.Slice {
+ panic(fmt.Errorf("AnySliceToInterfaceSlice() called on %T instead of a slice type", in))
+ }
+
+ out := make([]interface{}, v.Len())
+ for i := 0; i < v.Len(); i++ {
+ out[i] = v.Index(i).Interface()
+ }
+ return out
+}
+
+func SliceSubsets(in ...string) [][]string {
+ result := make([][]string, 0, 1<<len(in))
+
+ for bitset := 0; bitset < (1 << len(in)); bitset++ {
+ var subset []string
+ for i := 0; i < len(in); i++ {
+ if bitset&(1<<i) != 0 {
+ subset = append(subset, in[i])
+ }
+ }
+ result = append(result, subset)
+ }
+
+ return result
+}
diff --git a/tests/internal/utils/slice_test.go b/tests/internal/utils/slice_test.go
new file mode 100644
index 0000000..c878226
--- /dev/null
+++ b/tests/internal/utils/slice_test.go
@@ -0,0 +1,31 @@
+package utils
+
+import (
+ "github.com/stretchr/testify/require"
+ "testing"
+)
+
+func TestSliceSubsets(t *testing.T) {
+ data := []string{
+ "bla",
+ "blub",
+ "derp",
+ }
+
+ result := SliceSubsets(data...)
+
+ expected := [][]string{
+ nil,
+ {"bla"},
+ {"blub"},
+ {"bla", "blub"},
+ {"derp"},
+ {"bla", "derp"},
+ {"blub", "derp"},
+ {"bla", "blub", "derp"},
+ }
+
+ require.Equal(t, expected, result)
+
+ t.Logf("%#v", result)
+}
diff --git a/tests/internal/value/notification_states.go b/tests/internal/value/notification_states.go
new file mode 100644
index 0000000..3c5479e
--- /dev/null
+++ b/tests/internal/value/notification_states.go
@@ -0,0 +1,37 @@
+package value
+
+import "fmt"
+
+type NotificationStates []string
+
+func (s NotificationStates) IcingaDbValue() interface{} {
+ v := uint(0)
+
+ for _, s := range s {
+ if bit, ok := notificationStateMap[s]; ok {
+ v |= bit
+ } else {
+ panic(fmt.Errorf("unknown notification state %q", s))
+ }
+ }
+
+ return v
+}
+
+func (s NotificationStates) Icinga2ConfigValue() string {
+ return ToIcinga2Config([]string(s))
+}
+
+func (s NotificationStates) Icinga2ApiValue() interface{} {
+ return ToIcinga2Api([]string(s))
+}
+
+// https://github.com/Icinga/icinga2/blob/a8f98cf72115d50152137bc924277b426f483a3f/lib/icinga/notification.hpp#L20-L32
+var notificationStateMap = map[string]uint{
+ "OK": 1,
+ "Warning": 2,
+ "Critical": 4,
+ "Unknown": 8,
+ "Up": 16,
+ "Down": 32,
+}
diff --git a/tests/internal/value/notification_types.go b/tests/internal/value/notification_types.go
new file mode 100644
index 0000000..f368ed5
--- /dev/null
+++ b/tests/internal/value/notification_types.go
@@ -0,0 +1,40 @@
+package value
+
+import "fmt"
+
+type NotificationTypes []string
+
+func (t NotificationTypes) IcingaDbValue() interface{} {
+ v := uint(0)
+
+ for _, s := range t {
+ if bit, ok := notificationTypeMap[s]; ok {
+ v |= bit
+ } else {
+ panic(fmt.Errorf("unknown notification type %q", s))
+ }
+ }
+
+ return v
+}
+
+func (t NotificationTypes) Icinga2ConfigValue() string {
+ return ToIcinga2Config([]string(t))
+}
+
+func (t NotificationTypes) Icinga2ApiValue() interface{} {
+ return ToIcinga2Api([]string(t))
+}
+
+// https://github.com/Icinga/icinga2/blob/a8f98cf72115d50152137bc924277b426f483a3f/lib/icinga/notification.hpp#L34-L50
+var notificationTypeMap = map[string]uint{
+ "DowntimeStart": 1,
+ "DowntimeEnd": 2,
+ "DowntimeRemoved": 4,
+ "Custom": 8,
+ "Acknowledgement": 16,
+ "Problem": 32,
+ "Recovery": 64,
+ "FlappingStart": 128,
+ "FlappingEnd": 256,
+}
diff --git a/tests/internal/value/value.go b/tests/internal/value/value.go
new file mode 100644
index 0000000..87027dd
--- /dev/null
+++ b/tests/internal/value/value.go
@@ -0,0 +1,103 @@
+package value
+
+import (
+ "fmt"
+ "reflect"
+)
+
+func ToIcinga2Config(value interface{}) string {
+ if value == nil {
+ return "null"
+ }
+
+ refVal := reflect.ValueOf(value)
+ switch refVal.Kind() {
+ case reflect.Slice:
+ vs := ""
+ for i := 0; i < refVal.Len(); i++ {
+ vs += ToIcinga2Config(refVal.Index(i).Interface()) + ","
+ }
+ return "[" + vs + "]"
+ case reflect.Map:
+ kvs := ""
+ iter := refVal.MapRange()
+ for iter.Next() {
+ kvs += ToIcinga2Config(iter.Key().Interface()) + "=" + ToIcinga2Config(iter.Value().Interface()) + ","
+ }
+ return "{" + kvs + "}"
+ }
+
+ switch v := value.(type) {
+ case interface{ Icinga2ConfigValue() string }:
+ return v.Icinga2ConfigValue()
+ case string:
+ // TODO(jb): probably not perfect quoting, but good enough for now
+ return fmt.Sprintf("%q", v)
+ case *string:
+ if v != nil {
+ return ToIcinga2Config(*v)
+ } else {
+ return "null"
+ }
+ case bool, int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, float32, float64:
+ return fmt.Sprintf("%v", v)
+ default:
+ panic(fmt.Errorf("ToIcinga2Config called on unknown type %T", value))
+ }
+}
+
+func ToIcinga2Api(value interface{}) interface{} {
+ if value == nil {
+ return nil
+ }
+
+ refVal := reflect.ValueOf(value)
+ switch refVal.Kind() {
+ case reflect.Slice:
+ r := make([]interface{}, refVal.Len())
+ for i := range r {
+ r[i] = ToIcinga2Api(refVal.Index(i).Interface())
+ }
+ return r
+ case reflect.Map:
+ r := make(map[string]interface{})
+ iter := refVal.MapRange()
+ for iter.Next() {
+ // TODO: perform a better check than the type assertion
+ r[ToIcinga2Api(iter.Key().Interface()).(string)] = ToIcinga2Api(iter.Value().Interface())
+ }
+ return r
+ }
+
+ switch v := value.(type) {
+ case interface{ Icinga2ApiValue() interface{} }:
+ return v.Icinga2ApiValue()
+ case string, []string, bool, int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, float32, float64:
+ return v
+ case *string:
+ if v != nil {
+ return ToIcinga2Api(*v)
+ } else {
+ return nil
+ }
+ default:
+ panic(fmt.Errorf("ToIcinga2Api called on unknown type %T", value))
+ }
+}
+
+func ToIcingaDb(value interface{}) interface{} {
+ switch v := value.(type) {
+ case interface{ IcingaDbValue() interface{} }:
+ return v.IcingaDbValue()
+ case bool:
+ if v {
+ return "y"
+ } else {
+ return "n"
+ }
+ case string, *string, int, int8, int16, int32, int64, uint, uint8, uint16, uint32, uint64, float32, float64:
+ return v
+ default:
+ panic(fmt.Errorf("ToIcinga2Api called on unknown type %T", value))
+ }
+}