Skip to content

Commit 7879f9c

Browse files
authored
feat: Fix Contains to work also with slices (#5)
1 parent 24ef1e5 commit 7879f9c

File tree

3 files changed

+83
-14
lines changed

3 files changed

+83
-14
lines changed

test.go

+41-10
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package test
22

33
import (
4+
"fmt"
45
"reflect"
56
"strings"
67
"testing"
@@ -108,22 +109,52 @@ func False(t testing.TB, value bool) {
108109
}
109110
}
110111

111-
// Contains calls t.Fatalf if needle is not contained in the string haystack.
112-
func Contains[S ~string](t testing.TB, haystack S, needle S) {
112+
// Contains calls t.Fatalf if needle is not contained in haystack.
113+
// H can be either a string type (including custom string types) or a slice of comparable type N.
114+
// When H is a string type and N is any type, fmt.Sprintf is used to convert N to string for comparison.
115+
// When H is a slice, N is same type as the slice elements for direct comparison.
116+
func Contains[H interface{ ~string | []N }, N comparable](t testing.TB, haystack H, needle N) {
113117
t.Helper()
114-
if !contains(haystack, needle) {
115-
t.Fatalf("%q not in %q", needle, haystack)
118+
msg, found := containsElement(haystack, needle)
119+
// TODO: Fix this to use t.Fatal
120+
if !found {
121+
t.Fatalf("%s", msg)
116122
}
117123
}
118124

119-
// NotContains calls t.Fatalf if needle is contained in the string haystack.
120-
func NotContains[S ~string](t testing.TB, haystack S, needle S) {
125+
// NotContains calls t.Fatalf if needle is contained in haystack.
126+
// For type of H, N see [Contains]
127+
func NotContains[H interface{ ~string | []N }, N comparable](t testing.TB, haystack H, needle N) {
121128
t.Helper()
122-
if contains(haystack, needle) {
123-
t.Fatalf("%q in %q", needle, haystack)
129+
msg, found := containsElement(haystack, needle)
130+
// TODO: Fix this to use t.Fatal
131+
if found {
132+
t.Fatalf("%s", msg)
124133
}
125134
}
126135

127-
func contains[S ~string](haystack S, needle S) bool {
128-
return strings.Contains(string(haystack), string(needle))
136+
func containsElement[H interface{ ~string | []N }, N comparable](haystack H, needle N) (string, bool) {
137+
switch h := any(haystack).(type) {
138+
case string:
139+
n := fmt.Sprintf("%v", needle)
140+
if strings.Contains(h, n) {
141+
return fmt.Sprintf("%q in %q", n, h), true
142+
}
143+
return fmt.Sprintf("%q not in %q", n, h), false
144+
case []N:
145+
// TODO: refactor this using slices.Contains
146+
for _, v := range h {
147+
if v == needle {
148+
return fmt.Sprintf("%v in %v", needle, haystack), true
149+
}
150+
}
151+
return fmt.Sprintf("%v not in %v", needle, haystack), false
152+
default: // h is custom string type
153+
hs := fmt.Sprintf("%v", haystack)
154+
n := fmt.Sprintf("%v", needle)
155+
if strings.Contains(hs, n) {
156+
return fmt.Sprintf("%q in %q", n, hs), true
157+
}
158+
return fmt.Sprintf("%q not in %q", n, hs), false
159+
}
129160
}

test_example_test.go

+10-4
Original file line numberDiff line numberDiff line change
@@ -32,11 +32,15 @@ func Example() {
3232

3333
type mytype string
3434
var mystring mytype = "hello, world"
35-
test.Contains(t, mystring, "world") // good
36-
test.Contains(t, mystring, "World") // bad
35+
test.Contains(t, mystring, "world") // good
36+
test.Contains(t, mystring, "World") // bad
37+
test.Contains(t, []int{1, 2, 3, 4, 5}, 3) // good
38+
test.Contains(t, []int{1, 2, 3, 4, 5}, 6) // bad
3739

38-
test.NotContains(t, mystring, "World") // good
39-
test.NotContains(t, mystring, "world") // bad
40+
test.NotContains(t, mystring, "World") // good
41+
test.NotContains(t, mystring, "world") // bad
42+
test.NotContains(t, []int{1, 2, 3, 4, 5}, 6) // good
43+
test.NotContains(t, []int{1, 2, 3, 4, 5}, 3) // bad
4044

4145
// Output:
4246
// want: hello; got: world
@@ -46,5 +50,7 @@ func Example() {
4650
// got: <nil>
4751
// got: (O_o)
4852
// "World" not in "hello, world"
53+
// 6 not in [1 2 3 4 5]
4954
// "world" in "hello, world"
55+
// 3 in [1 2 3 4 5]
5056
}

test_test.go

+32
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,9 @@ func Test(t *testing.T) {
3434
beOkay(func(tb testing.TB) { test.True(tb, true) })
3535
beOkay(func(tb testing.TB) { test.False(tb, false) })
3636
beOkay(func(tb testing.TB) { test.Contains(tb, "hello world", "world") })
37+
beOkay(func(tb testing.TB) { test.Contains(t, []int{1, 2, 3, 4, 5}, 3) })
3738
beOkay(func(tb testing.TB) { test.NotContains(tb, "hello world", "World") })
39+
beOkay(func(tb testing.TB) { test.NotContains(t, []int{1, 2, 3, 4, 5}, 6) })
3840
beBad := func(callback func(tb testing.TB)) {
3941
t.Helper()
4042
var buf strings.Builder
@@ -56,7 +58,37 @@ func Test(t *testing.T) {
5658
beBad(func(tb testing.TB) { test.True(tb, false) })
5759
beBad(func(tb testing.TB) { test.False(tb, true) })
5860
beBad(func(tb testing.TB) { test.Contains(tb, "hello world", "World") })
61+
beBad(func(tb testing.TB) { test.Contains(tb, []int{1, 2, 3, 4, 5}, 6) })
5962
beBad(func(tb testing.TB) { test.NotContains(tb, "hello world", "world") })
63+
beBad(func(tb testing.TB) { test.NotContains(tb, []int{1, 2, 3, 4, 5}, 3) })
64+
}
65+
66+
func TestContains(t *testing.T) {
67+
// Case 1: String containment - when first parameter is string
68+
// The second parameter is automatically converted to string for comparison
69+
test.Contains(t, "3.141592", "3.14")
70+
test.Contains(t, "3.141592", 3) // Integer is converted to string
71+
test.Contains(t, "3.141592", 3.14) // Float is converted to string
72+
73+
// Case 2: Custom string type compatibility
74+
// Contains works with custom string types (~string) in any combination
75+
type customString string
76+
test.Contains(t, customString("abc"), customString("a"))
77+
test.Contains(t, customString("abc"), "a")
78+
test.Contains(t, "abc", customString("a"))
79+
80+
// Case 3: Slice element containment
81+
// When first parameter is a slice, Contains checks if the second parameter exists as an element
82+
test.Contains(t, []int{1, 2, 3, 4, 5}, 3)
83+
test.Contains(t, []string{"apple", "banana", "orange"}, "banana")
84+
test.Contains(t, []float64{1.1, 2.2, 3.3}, 2.2)
85+
test.Contains(t, []byte{1, 2, 3}, byte(2))
86+
87+
// Case 4: Custom type slice compatibility
88+
// Contains works with slices of any comparable type
89+
type customInt int
90+
nums := []customInt{1, 2, 3, 4}
91+
test.Contains(t, nums, customInt(2))
6092
}
6193

6294
type mockingT struct {

0 commit comments

Comments
 (0)