mirror of https://github.com/grafana/grafana.git
173 lines
4.4 KiB
Go
173 lines
4.4 KiB
Go
package util
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestIsInterfaceNil(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
value interface{}
|
|
expected bool
|
|
}{
|
|
// True nil cases
|
|
{"true nil interface", nil, true},
|
|
{"nil pointer", (*int)(nil), true},
|
|
{"nil slice", ([]int)(nil), true},
|
|
{"nil map", (map[string]int)(nil), true},
|
|
{"nil function", (func())(nil), true},
|
|
{"nil interface wrapped in interface", (interface{})(nil), true},
|
|
|
|
// Channels are not handled by IsInterfaceNil (not in switch statement)
|
|
{"nil channel - not handled", (chan int)(nil), false},
|
|
|
|
// Non-nil cases
|
|
{"non-nil pointer", func() interface{} { val := 42; return &val }(), false},
|
|
{"non-nil slice", []int{1, 2, 3}, false},
|
|
{"empty slice", []int{}, false},
|
|
{"non-nil map", map[string]int{"key": 1}, false},
|
|
{"empty map", make(map[string]int), false},
|
|
{"non-nil function", func() {}, false},
|
|
{"non-nil channel", make(chan int), false},
|
|
|
|
// Basic value types
|
|
{"string value", "hello", false},
|
|
{"empty string", "", false},
|
|
{"int value", 42, false},
|
|
{"zero int", 0, false},
|
|
{"bool true", true, false},
|
|
{"bool false", false, false},
|
|
{"float64", 3.14, false},
|
|
{"complex128", complex(1, 2), false},
|
|
|
|
// Composite value types
|
|
{"struct value", struct{ x int }{x: 1}, false},
|
|
{"array value", [3]int{1, 2, 3}, false},
|
|
{"zero array", [3]int{}, false},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
result := IsInterfaceNil(tc.value)
|
|
assert.Equal(t, tc.expected, result, "IsInterfaceNil(%v) should return %t", tc.value, tc.expected)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsInterfaceNil_NestedInterfaces(t *testing.T) {
|
|
testCases := []struct {
|
|
name string
|
|
value interface{}
|
|
expected bool
|
|
}{
|
|
{
|
|
name: "nested interface with nil",
|
|
value: func() interface{} {
|
|
var inner *int = nil
|
|
var middle interface{} = inner
|
|
return middle
|
|
}(),
|
|
expected: true,
|
|
},
|
|
{
|
|
name: "nested interface with value",
|
|
value: func() interface{} {
|
|
val := 42
|
|
inner := &val
|
|
var middle interface{} = inner
|
|
return middle
|
|
}(),
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "interface containing interface with value",
|
|
value: func() interface{} {
|
|
var inner interface{} = 42
|
|
outer := inner
|
|
return outer
|
|
}(),
|
|
expected: false,
|
|
},
|
|
{
|
|
name: "interface containing nil interface",
|
|
value: func() interface{} {
|
|
var inner interface{} = nil
|
|
outer := inner
|
|
return outer
|
|
}(),
|
|
expected: true,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
result := IsInterfaceNil(tc.value)
|
|
assert.Equal(t, tc.expected, result)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsInterfaceNil_ReflectKinds(t *testing.T) {
|
|
t.Run("handles specific nilable reflect kinds", func(t *testing.T) {
|
|
// Test the specific nilable kinds that the function handles
|
|
// according to its switch statement: Ptr, Slice, Map, Func, Interface
|
|
nilableTestCases := []struct {
|
|
name string
|
|
value interface{}
|
|
}{
|
|
{"nil pointer", (*int)(nil)},
|
|
{"nil slice", ([]int)(nil)},
|
|
{"nil map", (map[string]int)(nil)},
|
|
{"nil function", (func())(nil)},
|
|
{"nil interface", (interface{})(nil)},
|
|
}
|
|
|
|
for _, tc := range nilableTestCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
assert.True(t, IsInterfaceNil(tc.value), "%s should be detected as nil", tc.name)
|
|
})
|
|
}
|
|
})
|
|
|
|
t.Run("does not handle channels and other nilable types", func(t *testing.T) {
|
|
// Test that nilable kinds NOT in the switch statement return false
|
|
unhandledTestCases := []struct {
|
|
name string
|
|
value interface{}
|
|
}{
|
|
{"nil channel", (chan int)(nil)},
|
|
// UnsafePointer would be another example, but harder to test
|
|
}
|
|
|
|
for _, tc := range unhandledTestCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
assert.False(t, IsInterfaceNil(tc.value), "%s should not be detected as nil (unhandled type)", tc.name)
|
|
})
|
|
}
|
|
})
|
|
|
|
t.Run("handles non-nilable kinds", func(t *testing.T) {
|
|
// Test kinds that cannot be nil
|
|
nonNilableTestCases := []struct {
|
|
name string
|
|
value interface{}
|
|
}{
|
|
{"int", 42},
|
|
{"string", "test"},
|
|
{"bool", true},
|
|
{"struct", struct{ x int }{x: 1}},
|
|
{"array", [3]int{1, 2, 3}},
|
|
{"float64", 3.14},
|
|
{"complex128", complex(1, 2)},
|
|
}
|
|
|
|
for _, tc := range nonNilableTestCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
assert.False(t, IsInterfaceNil(tc.value), "%s should not be detected as nil (non-nilable)", tc.name)
|
|
})
|
|
}
|
|
})
|
|
}
|