diff --git a/README.md b/README.md
index 94483a4e..7b4a22eb 100644
--- a/README.md
+++ b/README.md
@@ -370,6 +370,8 @@ assert.Error(t, err)
```go
❌
assert.Equal(t, result, expected)
+assert.Equal(t, result, len(expected))
+assert.Equal(t, len(resultFields), len(expectedFields))
assert.EqualExportedValues(t, resultObj, User{Name: "Rob"})
assert.EqualValues(t, result, 42)
assert.Exactly(t, result, int64(42))
@@ -389,6 +391,8 @@ assert.YAMLEq(t, result, "version: '3'")
✅
assert.Equal(t, expected, result)
+assert.Equal(t, len(expected), result)
+assert.Equal(t, len(expectedFields), len(resultFields))
assert.EqualExportedValues(t, User{Name: "Rob"}, resultObj)
assert.EqualValues(t, 42, result)
// And so on...
@@ -618,19 +622,41 @@ P.S. Look at [testify's issue](https://github.com/stretchr/testify/issues/772),
```go
❌
-assert.Equal(t, 3, len(arr))
-assert.EqualValues(t, 3, len(arr))
-assert.Exactly(t, 3, len(arr))
-assert.True(t, len(arr) == 3)
+assert.Equal(t, 42, len(arr))
+assert.Equal(t, len(arr), 42)
+assert.EqualValues(t, 42, len(arr))
+assert.EqualValues(t, len(arr), 42)
+assert.Exactly(t, 42, len(arr))
+assert.Exactly(t, len(arr), 42)
+assert.True(t, 42 == len(arr))
+assert.True(t, len(arr) == 42)
+
+assert.Equal(t, value, len(arr))
+assert.EqualValues(t, value, len(arr))
+assert.Exactly(t, value, len(arr))
+assert.True(t, len(arr) == value)
+
+assert.Equal(t, len(expArr), len(arr))
+assert.EqualValues(t, len(expArr), len(arr))
+assert.Exactly(t, len(expArr), len(arr))
+assert.True(t, len(arr) == len(expArr))
✅
-assert.Len(t, arr, 3)
+assert.Len(t, arr, 42)
+assert.Len(t, arr, value)
+assert.Len(t, arr, len(expArr))
```
**Autofix**: true.
**Enabled by default**: true.
**Reason**: More appropriate `testify` API with clearer failure message.
+> [!CAUTION]
+> The checker ignores assertions in which length checking is not a priority, e.g.
+> ```go
+> assert.Equal(t, len(arr), value)
+> ```
+
---
### negative-positive
diff --git a/analyzer/checkers_factory_test.go b/analyzer/checkers_factory_test.go
index 28bfb1dd..ad25d664 100644
--- a/analyzer/checkers_factory_test.go
+++ b/analyzer/checkers_factory_test.go
@@ -18,7 +18,6 @@ func Test_newCheckers(t *testing.T) {
checkers.NewFloatCompare(),
checkers.NewBoolCompare(),
checkers.NewEmpty(),
- checkers.NewLen(),
checkers.NewNegativePositive(),
checkers.NewCompares(),
checkers.NewContains(),
@@ -27,6 +26,7 @@ func Test_newCheckers(t *testing.T) {
checkers.NewErrorIsAs(),
checkers.NewEncodedCompare(),
checkers.NewExpectedActual(),
+ checkers.NewLen(),
checkers.NewRegexp(),
checkers.NewSuiteExtraAssertCall(),
checkers.NewSuiteDontUsePkg(),
@@ -37,7 +37,6 @@ func Test_newCheckers(t *testing.T) {
checkers.NewFloatCompare(),
checkers.NewBoolCompare(),
checkers.NewEmpty(),
- checkers.NewLen(),
checkers.NewNegativePositive(),
checkers.NewCompares(),
checkers.NewContains(),
@@ -46,6 +45,7 @@ func Test_newCheckers(t *testing.T) {
checkers.NewErrorIsAs(),
checkers.NewEncodedCompare(),
checkers.NewExpectedActual(),
+ checkers.NewLen(),
checkers.NewRegexp(),
checkers.NewSuiteExtraAssertCall(),
checkers.NewSuiteDontUsePkg(),
diff --git a/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go b/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go
index 1a7aff00..160891d7 100644
--- a/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go
+++ b/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go
@@ -30,6 +30,7 @@ func TestExpectedActualChecker(t *testing.T) {
expectedVal := func() any { return nil }
var expectedObj struct{ Val int }
var expectedObjPtr = &expectedObj
+ var expectedFields []string
var result any
@@ -81,6 +82,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.Equalf(t, result, *(tt.expPtr()), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, result, ttPtr.expected) // want "expected-actual: need to reverse actual and expected values"
assert.Equalf(t, result, ttPtr.expected, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
+ assert.Equal(t, result, len(expectedFields)) // want "expected-actual: need to reverse actual and expected values"
+ assert.Equalf(t, result, len(expectedFields), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, result, 42) // want "expected-actual: need to reverse actual and expected values"
assert.Equalf(t, result, 42, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, result, 3.14) // want "expected-actual: need to reverse actual and expected values"
@@ -140,6 +143,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.NotEqualf(t, result, *(tt.expPtr()), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, result, ttPtr.expected) // want "expected-actual: need to reverse actual and expected values"
assert.NotEqualf(t, result, ttPtr.expected, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
+ assert.NotEqual(t, result, len(expectedFields)) // want "expected-actual: need to reverse actual and expected values"
+ assert.NotEqualf(t, result, len(expectedFields), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, result, 42) // want "expected-actual: need to reverse actual and expected values"
assert.NotEqualf(t, result, 42, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, result, 3.14) // want "expected-actual: need to reverse actual and expected values"
@@ -202,6 +207,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.Equalf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42")
assert.Equal(t, ttPtr.expected, result)
assert.Equalf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(expectedFields), result)
+ assert.Equalf(t, len(expectedFields), result, "msg with args %d %s", 42, "42")
assert.Equal(t, 42, result)
assert.Equalf(t, 42, result, "msg with args %d %s", 42, "42")
assert.Equal(t, 3.14, result)
@@ -261,6 +268,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.NotEqualf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, ttPtr.expected, result)
assert.NotEqualf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42")
+ assert.NotEqual(t, len(expectedFields), result)
+ assert.NotEqualf(t, len(expectedFields), result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, 42, result)
assert.NotEqualf(t, 42, result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, 3.14, result)
@@ -278,15 +287,18 @@ func TestExpectedActualChecker(t *testing.T) {
func TestExpectedActualChecker_Other(t *testing.T) {
var (
- result, expected any
- resultPtr, expectedPtr *int
- resultObj, expectedObj user
- resultTime, expectedTime time.Time
- value int
+ result, expected any
+ resultPtr, expectedPtr *int
+ resultObj, expectedObj user
+ resultTime, expectedTime time.Time
+ value int
+ actualFields, expectedFields []string
)
// Invalid.
{
+ assert.Equal(t, len(actualFields), len(expectedFields)) // want "expected-actual: need to reverse actual and expected values"
+ assert.Equalf(t, len(actualFields), len(expectedFields), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValues(t, resultObj, expectedObj) // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValuesf(t, resultObj, expectedObj, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValues(t, resultObj, user{Name: "Rob"}) // want "expected-actual: need to reverse actual and expected values"
@@ -1150,10 +1162,11 @@ func TestExpectedActualChecker_CannotDetectVariablesLookedLikeConsts(t *testing.
func TestExpectedActualChecker_Ignored(t *testing.T) {
var (
- result, expected any
- resultPtr, expectedPtr *int
- value int
- expectedTime time.Time
+ result, expected any
+ resultPtr, expectedPtr *int
+ value int
+ expectedTime time.Time
+ expectedFields, interfaces, directions []string
)
assert.Equal(t, nil, nil)
@@ -1162,6 +1175,10 @@ func TestExpectedActualChecker_Ignored(t *testing.T) {
assert.Equalf(t, "value", "value", "msg with args %d %s", 42, "42")
assert.Equal(t, expected, expected)
assert.Equalf(t, expected, expected, "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(expectedFields), len(expectedFields))
+ assert.Equalf(t, len(expectedFields), len(expectedFields), "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(interfaces), len(directions))
+ assert.Equalf(t, len(interfaces), len(directions), "msg with args %d %s", 42, "42")
assert.Equal(t, value, &resultPtr)
assert.Equalf(t, value, &resultPtr, "msg with args %d %s", 42, "42")
assert.Equal(t, []int{1, 2}, map[int]int{1: 2})
diff --git a/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go.golden b/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go.golden
index b0a37bd7..b059f493 100644
--- a/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go.golden
+++ b/analyzer/testdata/src/checkers-default/expected-actual/expected_actual_test.go.golden
@@ -30,6 +30,7 @@ func TestExpectedActualChecker(t *testing.T) {
expectedVal := func() any { return nil }
var expectedObj struct{ Val int }
var expectedObjPtr = &expectedObj
+ var expectedFields []string
var result any
@@ -81,6 +82,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.Equalf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, ttPtr.expected, result) // want "expected-actual: need to reverse actual and expected values"
assert.Equalf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
+ assert.Equal(t, len(expectedFields), result) // want "expected-actual: need to reverse actual and expected values"
+ assert.Equalf(t, len(expectedFields), result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, 42, result) // want "expected-actual: need to reverse actual and expected values"
assert.Equalf(t, 42, result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.Equal(t, 3.14, result) // want "expected-actual: need to reverse actual and expected values"
@@ -140,6 +143,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.NotEqualf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, ttPtr.expected, result) // want "expected-actual: need to reverse actual and expected values"
assert.NotEqualf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
+ assert.NotEqual(t, len(expectedFields), result) // want "expected-actual: need to reverse actual and expected values"
+ assert.NotEqualf(t, len(expectedFields), result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, 42, result) // want "expected-actual: need to reverse actual and expected values"
assert.NotEqualf(t, 42, result, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.NotEqual(t, 3.14, result) // want "expected-actual: need to reverse actual and expected values"
@@ -202,6 +207,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.Equalf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42")
assert.Equal(t, ttPtr.expected, result)
assert.Equalf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(expectedFields), result)
+ assert.Equalf(t, len(expectedFields), result, "msg with args %d %s", 42, "42")
assert.Equal(t, 42, result)
assert.Equalf(t, 42, result, "msg with args %d %s", 42, "42")
assert.Equal(t, 3.14, result)
@@ -261,6 +268,8 @@ func TestExpectedActualChecker(t *testing.T) {
assert.NotEqualf(t, *(tt.expPtr()), result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, ttPtr.expected, result)
assert.NotEqualf(t, ttPtr.expected, result, "msg with args %d %s", 42, "42")
+ assert.NotEqual(t, len(expectedFields), result)
+ assert.NotEqualf(t, len(expectedFields), result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, 42, result)
assert.NotEqualf(t, 42, result, "msg with args %d %s", 42, "42")
assert.NotEqual(t, 3.14, result)
@@ -278,15 +287,18 @@ func TestExpectedActualChecker(t *testing.T) {
func TestExpectedActualChecker_Other(t *testing.T) {
var (
- result, expected any
- resultPtr, expectedPtr *int
- resultObj, expectedObj user
- resultTime, expectedTime time.Time
- value int
+ result, expected any
+ resultPtr, expectedPtr *int
+ resultObj, expectedObj user
+ resultTime, expectedTime time.Time
+ value int
+ actualFields, expectedFields []string
)
// Invalid.
{
+ assert.Equal(t, len(expectedFields), len(actualFields)) // want "expected-actual: need to reverse actual and expected values"
+ assert.Equalf(t, len(expectedFields), len(actualFields), "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValues(t, expectedObj, resultObj) // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValuesf(t, expectedObj, resultObj, "msg with args %d %s", 42, "42") // want "expected-actual: need to reverse actual and expected values"
assert.EqualExportedValues(t, user{Name: "Rob"}, resultObj) // want "expected-actual: need to reverse actual and expected values"
@@ -1150,10 +1162,11 @@ func TestExpectedActualChecker_CannotDetectVariablesLookedLikeConsts(t *testing.
func TestExpectedActualChecker_Ignored(t *testing.T) {
var (
- result, expected any
- resultPtr, expectedPtr *int
- value int
- expectedTime time.Time
+ result, expected any
+ resultPtr, expectedPtr *int
+ value int
+ expectedTime time.Time
+ expectedFields, interfaces, directions []string
)
assert.Equal(t, nil, nil)
@@ -1162,6 +1175,10 @@ func TestExpectedActualChecker_Ignored(t *testing.T) {
assert.Equalf(t, "value", "value", "msg with args %d %s", 42, "42")
assert.Equal(t, expected, expected)
assert.Equalf(t, expected, expected, "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(expectedFields), len(expectedFields))
+ assert.Equalf(t, len(expectedFields), len(expectedFields), "msg with args %d %s", 42, "42")
+ assert.Equal(t, len(interfaces), len(directions))
+ assert.Equalf(t, len(interfaces), len(directions), "msg with args %d %s", 42, "42")
assert.Equal(t, value, &resultPtr)
assert.Equalf(t, value, &resultPtr, "msg with args %d %s", 42, "42")
assert.Equal(t, []int{1, 2}, map[int]int{1: 2})
diff --git a/analyzer/testdata/src/checkers-default/len/len_test.go b/analyzer/testdata/src/checkers-default/len/len_test.go
index 8f1b3ecb..4a42a859 100644
--- a/analyzer/testdata/src/checkers-default/len/len_test.go
+++ b/analyzer/testdata/src/checkers-default/len/len_test.go
@@ -8,62 +8,74 @@ import (
"github.com/stretchr/testify/assert"
)
+const constNum = 10
+
func TestLenChecker(t *testing.T) {
- var arr [3]int
+ var arr, expArr [3]int
var value int
// Invalid.
{
- assert.Equal(t, len(arr), 42) // want "len: use assert\\.Len"
- assert.Equalf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Equal(t, 42, len(arr)) // want "len: use assert\\.Len"
- assert.Equalf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Equal(t, value, len(arr)) // want "len: use assert\\.Len"
- assert.Equalf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.EqualValues(t, len(arr), 42) // want "len: use assert\\.Len"
- assert.EqualValuesf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.EqualValues(t, 42, len(arr)) // want "len: use assert\\.Len"
- assert.EqualValuesf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.EqualValues(t, value, len(arr)) // want "len: use assert\\.Len"
- assert.EqualValuesf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Exactly(t, len(arr), 42) // want "len: use assert\\.Len"
- assert.Exactlyf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Exactly(t, 42, len(arr)) // want "len: use assert\\.Len"
- assert.Exactlyf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Exactly(t, value, len(arr)) // want "len: use assert\\.Len"
- assert.Exactlyf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.True(t, len(arr) == 42) // want "len: use assert\\.Len"
- assert.Truef(t, len(arr) == 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.True(t, 42 == len(arr)) // want "len: use assert\\.Len"
- assert.Truef(t, 42 == len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Equal(t, len(arr), 42) // want "len: use assert\\.Len"
+ assert.Equalf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Equal(t, 42, len(arr)) // want "len: use assert\\.Len"
+ assert.Equalf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Equal(t, value, len(arr)) // want "len: use assert\\.Len"
+ assert.Equalf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Equal(t, len(expArr), len(arr)) // want "len: use assert\\.Len"
+ assert.Equalf(t, len(expArr), len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.EqualValues(t, len(arr), 42) // want "len: use assert\\.Len"
+ assert.EqualValuesf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.EqualValues(t, 42, len(arr)) // want "len: use assert\\.Len"
+ assert.EqualValuesf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.EqualValues(t, value, len(arr)) // want "len: use assert\\.Len"
+ assert.EqualValuesf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.EqualValues(t, len(expArr), len(arr)) // want "len: use assert\\.Len"
+ assert.EqualValuesf(t, len(expArr), len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Exactly(t, len(arr), 42) // want "len: use assert\\.Len"
+ assert.Exactlyf(t, len(arr), 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Exactly(t, 42, len(arr)) // want "len: use assert\\.Len"
+ assert.Exactlyf(t, 42, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Exactly(t, value, len(arr)) // want "len: use assert\\.Len"
+ assert.Exactlyf(t, value, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Exactly(t, len(expArr), len(arr)) // want "len: use assert\\.Len"
+ assert.Exactlyf(t, len(expArr), len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.True(t, len(arr) == 42) // want "len: use assert\\.Len"
+ assert.Truef(t, len(arr) == 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.True(t, 42 == len(arr)) // want "len: use assert\\.Len"
+ assert.Truef(t, 42 == len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.True(t, len(arr) == value) // want "len: use assert\\.Len"
+ assert.Truef(t, len(arr) == value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.True(t, len(arr) == len(expArr)) // want "len: use assert\\.Len"
+ assert.Truef(t, len(arr) == len(expArr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Equal(t, constNum, len(arr)) // want "len: use assert\\.Len"
+ assert.Equalf(t, constNum, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.EqualValues(t, constNum, len(arr)) // want "len: use assert\\.Len"
+ assert.EqualValuesf(t, constNum, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Exactly(t, constNum, len(arr)) // want "len: use assert\\.Len"
+ assert.Exactlyf(t, constNum, len(arr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.True(t, len(arr) == constNum) // want "len: use assert\\.Len"
+ assert.Truef(t, len(arr) == constNum, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
}
// Valid.
{
assert.Len(t, arr, 42)
assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42")
+ assert.Len(t, arr, value)
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42")
assert.Len(t, arr, len(arr))
assert.Lenf(t, arr, len(arr), "msg with args %d %s", 42, "42")
}
// Ignored.
{
- assert.Equal(t, len(arr), len(arr))
- assert.Equalf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.Equal(t, len(arr), value)
assert.Equalf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.EqualValues(t, len(arr), len(arr))
- assert.EqualValuesf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.EqualValues(t, len(arr), value)
assert.EqualValuesf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.Exactly(t, len(arr), len(arr))
- assert.Exactlyf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.Exactly(t, len(arr), value)
assert.Exactlyf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.True(t, len(arr) == len(arr))
- assert.Truef(t, len(arr) == len(arr), "msg with args %d %s", 42, "42")
- assert.True(t, len(arr) == value)
- assert.Truef(t, len(arr) == value, "msg with args %d %s", 42, "42")
assert.True(t, value == len(arr))
assert.Truef(t, value == len(arr), "msg with args %d %s", 42, "42")
assert.NotEqual(t, 42, len(arr))
diff --git a/analyzer/testdata/src/checkers-default/len/len_test.go.golden b/analyzer/testdata/src/checkers-default/len/len_test.go.golden
index 9a0928e1..111395c7 100644
--- a/analyzer/testdata/src/checkers-default/len/len_test.go.golden
+++ b/analyzer/testdata/src/checkers-default/len/len_test.go.golden
@@ -8,62 +8,74 @@ import (
"github.com/stretchr/testify/assert"
)
+const constNum = 10
+
func TestLenChecker(t *testing.T) {
- var arr [3]int
+ var arr, expArr [3]int
var value int
// Invalid.
{
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, value) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, value) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, value) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
- assert.Len(t, arr, 42) // want "len: use assert\\.Len"
- assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, value) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, len(expArr)) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, len(expArr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, value) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, len(expArr)) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, len(expArr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, value) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, len(expArr)) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, len(expArr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, 42) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, value) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, len(expArr)) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, len(expArr), "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, constNum) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, constNum, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, constNum) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, constNum, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, constNum) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, constNum, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
+ assert.Len(t, arr, constNum) // want "len: use assert\\.Len"
+ assert.Lenf(t, arr, constNum, "msg with args %d %s", 42, "42") // want "len: use assert\\.Lenf"
}
// Valid.
{
assert.Len(t, arr, 42)
assert.Lenf(t, arr, 42, "msg with args %d %s", 42, "42")
+ assert.Len(t, arr, value)
+ assert.Lenf(t, arr, value, "msg with args %d %s", 42, "42")
assert.Len(t, arr, len(arr))
assert.Lenf(t, arr, len(arr), "msg with args %d %s", 42, "42")
}
// Ignored.
{
- assert.Equal(t, len(arr), len(arr))
- assert.Equalf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.Equal(t, len(arr), value)
assert.Equalf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.EqualValues(t, len(arr), len(arr))
- assert.EqualValuesf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.EqualValues(t, len(arr), value)
assert.EqualValuesf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.Exactly(t, len(arr), len(arr))
- assert.Exactlyf(t, len(arr), len(arr), "msg with args %d %s", 42, "42")
assert.Exactly(t, len(arr), value)
assert.Exactlyf(t, len(arr), value, "msg with args %d %s", 42, "42")
- assert.True(t, len(arr) == len(arr))
- assert.Truef(t, len(arr) == len(arr), "msg with args %d %s", 42, "42")
- assert.True(t, len(arr) == value)
- assert.Truef(t, len(arr) == value, "msg with args %d %s", 42, "42")
assert.True(t, value == len(arr))
assert.Truef(t, value == len(arr), "msg with args %d %s", 42, "42")
assert.NotEqual(t, 42, len(arr))
diff --git a/analyzer/testdata/src/checkers-priority/checkers_priority_test.go b/analyzer/testdata/src/checkers-priority/checkers_priority_test.go
index 1a6bb110..a2fa7fac 100644
--- a/analyzer/testdata/src/checkers-priority/checkers_priority_test.go
+++ b/analyzer/testdata/src/checkers-priority/checkers_priority_test.go
@@ -11,9 +11,9 @@ func TestCheckersPriority(t *testing.T) {
var f float64
var b bool
- // `empty` > `len` > `expected-actual`
+ // `empty` > `expected-actual` > `len`
assert.Equal(t, len([]int{}), 0) // want "empty: use assert\\.Empty"
- assert.Equal(t, len([]int{}), 3) // want "len: use assert\\.Len"
+ assert.Equal(t, len([]int{}), 3) // want "expected-actual: need to reverse actual and expected values"
// `float-compare` > `bool-compare` > `compares` > `expected-actual`
require.True(t, 42.42 == f) // want "float-compare: use require\\.InEpsilon \\(or InDelta\\)"
diff --git a/analyzer/testdata/src/checkers-priority/checkers_priority_test.go.golden b/analyzer/testdata/src/checkers-priority/checkers_priority_test.go.golden
index 3f2d8df7..983bb828 100644
--- a/analyzer/testdata/src/checkers-priority/checkers_priority_test.go.golden
+++ b/analyzer/testdata/src/checkers-priority/checkers_priority_test.go.golden
@@ -11,9 +11,9 @@ func TestCheckersPriority(t *testing.T) {
var f float64
var b bool
- // `empty` > `len` > `expected-actual`
- assert.Empty(t, []int{}) // want "empty: use assert\\.Empty"
- assert.Len(t, []int{}, 3) // want "len: use assert\\.Len"
+ // `empty` > `expected-actual` > `len`
+ assert.Empty(t, []int{}) // want "empty: use assert\\.Empty"
+ assert.Equal(t, 3, len([]int{})) // want "expected-actual: need to reverse actual and expected values"
// `float-compare` > `bool-compare` > `compares` > `expected-actual`
require.True(t, 42.42 == f) // want "float-compare: use require\\.InEpsilon \\(or InDelta\\)"
diff --git a/analyzer/testdata/src/debug/len_interface_test.go b/analyzer/testdata/src/debug/len_interface_test.go
new file mode 100644
index 00000000..1c37f95a
--- /dev/null
+++ b/analyzer/testdata/src/debug/len_interface_test.go
@@ -0,0 +1,26 @@
+package debug
+
+import (
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+)
+
+func TestLenInterface(t *testing.T) {
+ const n = 10
+ a := newArr(n)
+ assert.Equal(t, n, a.Len())
+ assert.Len(t, a, n) // Error: "{[ ]}" could not be applied builtin len()
+}
+
+type arr struct {
+ v []string
+}
+
+func newArr(n int) arr {
+ return arr{v: make([]string, n)}
+}
+
+func (a arr) Len() int {
+ return len(a.v)
+}
diff --git a/analyzer/testdata/src/debug/len_type_conversions_test.go b/analyzer/testdata/src/debug/len_type_conversions_test.go
new file mode 100644
index 00000000..21040f46
--- /dev/null
+++ b/analyzer/testdata/src/debug/len_type_conversions_test.go
@@ -0,0 +1,32 @@
+package debug
+
+import (
+ "encoding/json"
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+)
+
+func TestLenTypeConversions(t *testing.T) {
+ const resp = "Multi-языковая string 你好世界 🙂"
+ const respLen = len(resp) // 47
+
+ assert.Equal(t, respLen, len(resp))
+ assert.Equal(t, respLen, len([]byte(resp)))
+ assert.Equal(t, respLen, len(json.RawMessage(resp)))
+ assert.Len(t, resp, respLen)
+ assert.Len(t, []byte(resp), respLen)
+ assert.Len(t, json.RawMessage(resp), respLen)
+}
+
+func TestBytesReadability(t *testing.T) {
+ resp := []byte(`{"status": 200}`)
+ assert.Len(t, resp, 3) // "[123 34 115 116 97 116 117 115 34 58 32 50 48 48 125]" should have 3 item(s), but has 15
+ assert.Len(t, string(resp), 3) // "{"status": 200}" should have 3 item(s), but has 15
+
+ var noResp []byte
+ assert.Empty(t, resp) // Should be empty, but was [123 34 115 116 97 116 117 115 34 58 32 50 48 48 125]
+ assert.Empty(t, string(resp)) // Should be empty, but was {"status": 200}
+ assert.NotEmpty(t, noResp) // Should NOT be empty, but was []
+ assert.NotEmpty(t, string(noResp)) // Should NOT be empty, but was
+}
diff --git a/internal/checkers/checkers_registry.go b/internal/checkers/checkers_registry.go
index f881be4f..bb4aa4d0 100644
--- a/internal/checkers/checkers_registry.go
+++ b/internal/checkers/checkers_registry.go
@@ -10,7 +10,6 @@ var registry = checkersRegistry{
{factory: asCheckerFactory(NewFloatCompare), enabledByDefault: true},
{factory: asCheckerFactory(NewBoolCompare), enabledByDefault: true},
{factory: asCheckerFactory(NewEmpty), enabledByDefault: true},
- {factory: asCheckerFactory(NewLen), enabledByDefault: true},
{factory: asCheckerFactory(NewNegativePositive), enabledByDefault: true},
{factory: asCheckerFactory(NewCompares), enabledByDefault: true},
{factory: asCheckerFactory(NewContains), enabledByDefault: true},
@@ -19,6 +18,7 @@ var registry = checkersRegistry{
{factory: asCheckerFactory(NewErrorIsAs), enabledByDefault: true},
{factory: asCheckerFactory(NewEncodedCompare), enabledByDefault: true},
{factory: asCheckerFactory(NewExpectedActual), enabledByDefault: true},
+ {factory: asCheckerFactory(NewLen), enabledByDefault: true},
{factory: asCheckerFactory(NewRegexp), enabledByDefault: true},
{factory: asCheckerFactory(NewSuiteExtraAssertCall), enabledByDefault: true},
{factory: asCheckerFactory(NewSuiteDontUsePkg), enabledByDefault: true},
diff --git a/internal/checkers/checkers_registry_test.go b/internal/checkers/checkers_registry_test.go
index 08c3a397..eff55712 100644
--- a/internal/checkers/checkers_registry_test.go
+++ b/internal/checkers/checkers_registry_test.go
@@ -38,7 +38,6 @@ func TestAll(t *testing.T) {
"float-compare",
"bool-compare",
"empty",
- "len",
"negative-positive",
"compares",
"contains",
@@ -47,6 +46,7 @@ func TestAll(t *testing.T) {
"error-is-as",
"encoded-compare",
"expected-actual",
+ "len",
"regexp",
"suite-extra-assert-call",
"suite-dont-use-pkg",
@@ -75,7 +75,6 @@ func TestEnabledByDefault(t *testing.T) {
"float-compare",
"bool-compare",
"empty",
- "len",
"negative-positive",
"compares",
"contains",
@@ -84,6 +83,7 @@ func TestEnabledByDefault(t *testing.T) {
"error-is-as",
"encoded-compare",
"expected-actual",
+ "len",
"regexp",
"suite-extra-assert-call",
"suite-dont-use-pkg",
diff --git a/internal/checkers/expected_actual.go b/internal/checkers/expected_actual.go
index 351d675c..31ea3ff4 100644
--- a/internal/checkers/expected_actual.go
+++ b/internal/checkers/expected_actual.go
@@ -17,6 +17,8 @@ var DefaultExpectedVarPattern = regexp.MustCompile(
// ExpectedActual detects situations like
//
// assert.Equal(t, result, expected)
+// assert.Equal(t, result, len(expected))
+// assert.Equal(t, len(resultFields), len(expectedFields))
// assert.EqualExportedValues(t, resultObj, User{Name: "Anton"})
// assert.EqualValues(t, result, 42)
// assert.Exactly(t, result, int64(42))
@@ -37,6 +39,8 @@ var DefaultExpectedVarPattern = regexp.MustCompile(
// and requires
//
// assert.Equal(t, expected, result)
+// assert.Equal(t, len(expected), result)
+// assert.Equal(t, len(expectedFields), len(resultFields))
// assert.EqualExportedValues(t, User{Name: "Anton"}, resultObj)
// assert.EqualValues(t, 42, result)
// ...
@@ -122,6 +126,9 @@ func (checker ExpectedActual) isExpectedValueCandidate(pass *analysis.Pass, expr
return true
case *ast.CallExpr:
+ if lv, ok := isBuiltinLenCall(pass, expr); ok {
+ return isIdentNamedAfterPattern(checker.expVarPattern, lv)
+ }
return isParenExpr(v) ||
isCastedBasicLitOrExpectedValue(v, checker.expVarPattern) ||
isExpectedValueFactory(pass, v, checker.expVarPattern)
diff --git a/internal/checkers/helpers_len.go b/internal/checkers/helpers_len.go
index 904950ff..6afa5849 100644
--- a/internal/checkers/helpers_len.go
+++ b/internal/checkers/helpers_len.go
@@ -2,7 +2,6 @@ package checkers
import (
"go/ast"
- "go/token"
"go/types"
"golang.org/x/tools/go/analysis"
@@ -12,31 +11,6 @@ import (
var lenObj = types.Universe.Lookup("len")
-func isLenEquality(pass *analysis.Pass, e ast.Expr) (ast.Expr, ast.Expr, bool) {
- be, ok := e.(*ast.BinaryExpr)
- if !ok {
- return nil, nil, false
- }
-
- if be.Op != token.EQL {
- return nil, nil, false
- }
- return xorLenCall(pass, be.X, be.Y)
-}
-
-func xorLenCall(pass *analysis.Pass, a, b ast.Expr) (lenArg ast.Expr, expectedLen ast.Expr, ok bool) {
- arg1, ok1 := isBuiltinLenCall(pass, a)
- arg2, ok2 := isBuiltinLenCall(pass, b)
-
- if xor(ok1, ok2) {
- if ok1 {
- return arg1, b, true
- }
- return arg2, a, true
- }
- return nil, nil, false
-}
-
func isLenCallAndZero(pass *analysis.Pass, a, b ast.Expr) (ast.Expr, bool) {
lenArg, ok := isBuiltinLenCall(pass, a)
return lenArg, ok && isZero(b)
diff --git a/internal/checkers/len.go b/internal/checkers/len.go
index c240a617..9bdd8ff9 100644
--- a/internal/checkers/len.go
+++ b/internal/checkers/len.go
@@ -1,19 +1,42 @@
package checkers
import (
+ "go/ast"
+ "go/token"
+
"golang.org/x/tools/go/analysis"
)
// Len detects situations like
//
-// assert.Equal(t, 3, len(arr))
-// assert.EqualValues(t, 3, len(arr))
-// assert.Exactly(t, 3, len(arr))
-// assert.True(t, len(arr) == 3)
+// assert.Equal(t, 42, len(arr))
+// assert.Equal(t, len(arr), 42)
+// assert.EqualValues(t, 42, len(arr))
+// assert.EqualValues(t, len(arr), 42)
+// assert.Exactly(t, 42, len(arr))
+// assert.Exactly(t, len(arr), 42)
+// assert.True(t, 42 == len(arr))
+// assert.True(t, len(arr) == 42)
+//
+// assert.Equal(t, value, len(arr))
+// assert.EqualValues(t, value, len(arr))
+// assert.Exactly(t, value, len(arr))
+// assert.True(t, len(arr) == value)
+
+// assert.Equal(t, len(expArr), len(arr))
+// assert.EqualValues(t, len(expArr), len(arr))
+// assert.Exactly(t, len(expArr), len(arr))
+// assert.True(t, len(arr) == len(expArr))
//
// and requires
//
-// assert.Len(t, arr, 3)
+// assert.Len(t, arr, 42)
+// assert.Len(t, arr, value)
+// assert.Len(t, arr, len(expArr))
+//
+// The checker ignores assertions in which length checking is not a priority, e.g
+//
+// assert.Equal(t, len(arr), value)
type Len struct{}
// NewLen constructs Len checker.
@@ -21,45 +44,62 @@ func NewLen() Len { return Len{} }
func (Len) Name() string { return "len" }
func (checker Len) Check(pass *analysis.Pass, call *CallMeta) *analysis.Diagnostic {
- const proposedFn = "Len"
-
switch call.Fn.NameFTrimmed {
case "Equal", "EqualValues", "Exactly":
if len(call.Args) < 2 {
return nil
}
- a, b := call.Args[0], call.Args[1]
- if lenArg, expectedLen, ok := xorLenCall(pass, a, b); ok {
- if _, ok := isIntBasicLit(expectedLen); (expectedLen == b) && !ok {
- // https://github.com/Antonboom/testifylint/issues/9
- return nil
- }
- return newUseFunctionDiagnostic(checker.Name(), call, proposedFn,
- analysis.TextEdit{
- Pos: a.Pos(),
- End: b.End(),
- NewText: formatAsCallArgs(pass, lenArg, expectedLen),
- })
- }
+ a, b := call.Args[0], call.Args[1]
+ return checker.checkArgs(call, pass, a, b, false)
case "True":
if len(call.Args) < 1 {
return nil
}
- expr := call.Args[0]
- if lenArg, expectedLen, ok := isLenEquality(pass, expr); ok {
- if _, ok := isIntBasicLit(expectedLen); !ok {
- return nil
- }
- return newUseFunctionDiagnostic(checker.Name(), call, proposedFn,
- analysis.TextEdit{
- Pos: expr.Pos(),
- End: expr.End(),
- NewText: formatAsCallArgs(pass, lenArg, expectedLen),
- })
+ be, ok := call.Args[0].(*ast.BinaryExpr)
+ if !ok {
+ return nil
+ }
+ if be.Op != token.EQL {
+ return nil
+ }
+ return checker.checkArgs(call, pass, be.Y, be.X, true) // In True, the actual value is usually first.
+ }
+ return nil
+}
+
+func (checker Len) checkArgs(call *CallMeta, pass *analysis.Pass, a, b ast.Expr, inverted bool) *analysis.Diagnostic {
+ newUseLenDiagnostic := func(lenArg, expectedLen ast.Expr) *analysis.Diagnostic {
+ const proposedFn = "Len"
+ start, end := a.Pos(), b.End()
+ if inverted {
+ start, end = b.Pos(), a.End()
+ }
+ return newUseFunctionDiagnostic(checker.Name(), call, proposedFn,
+ analysis.TextEdit{
+ Pos: start,
+ End: end,
+ NewText: formatAsCallArgs(pass, lenArg, expectedLen),
+ })
+ }
+
+ arg1, firstIsLen := isBuiltinLenCall(pass, a)
+ arg2, secondIsLen := isBuiltinLenCall(pass, b)
+
+ switch {
+ case firstIsLen && secondIsLen:
+ return newUseLenDiagnostic(arg2, a)
+
+ case firstIsLen:
+ if _, secondIsNum := isIntBasicLit(b); secondIsNum {
+ return newUseLenDiagnostic(arg1, b)
}
+
+ case secondIsLen:
+ return newUseLenDiagnostic(arg2, a)
}
+
return nil
}
diff --git a/internal/testgen/gen_expected_actual.go b/internal/testgen/gen_expected_actual.go
index a969d9c3..708d5a2a 100644
--- a/internal/testgen/gen_expected_actual.go
+++ b/internal/testgen/gen_expected_actual.go
@@ -106,6 +106,8 @@ func (g ExpectedActualTestsGenerator) TemplateData() any {
"*(tt.expPtr())",
"ttPtr.expected",
+ "len(expectedFields)",
+
// NOTE(a.telyshev): Unsupported rare cases:
// "(*expectedObjPtr).Val",
// "(*ttPtr).expected",
@@ -122,6 +124,11 @@ func (g ExpectedActualTestsGenerator) TemplateData() any {
},
OtherExpActFunctions: test{
InvalidAssertions: []Assertion{
+ {
+ Fn: "Equal", Argsf: "len(actualFields), len(expectedFields)",
+ ReportMsgf: report, ProposedArgsf: "len(expectedFields), len(actualFields)",
+ },
+
{
Fn: "EqualExportedValues", Argsf: "resultObj, expectedObj",
ReportMsgf: report, ProposedArgsf: "expectedObj, resultObj",
@@ -261,6 +268,8 @@ func (g ExpectedActualTestsGenerator) TemplateData() any {
{Fn: "Equal", Argsf: "nil, nil"},
{Fn: "Equal", Argsf: `"value", "value"`},
{Fn: "Equal", Argsf: "expected, expected"},
+ {Fn: "Equal", Argsf: "len(expectedFields), len(expectedFields)"},
+ {Fn: "Equal", Argsf: "len(interfaces), len(directions)"},
{Fn: "Equal", Argsf: "value, &resultPtr"},
{Fn: "Equal", Argsf: "[]int{1, 2}, map[int]int{1: 2}"},
{Fn: "NotEqual", Argsf: "result, result"},
@@ -348,6 +357,7 @@ func {{ .CheckerName.AsTestName }}(t *testing.T) {
expectedVal := func() any { return nil }
var expectedObj struct { Val int }
var expectedObjPtr = &expectedObj
+ var expectedFields []string
var result any
@@ -369,6 +379,7 @@ func {{ .CheckerName.AsTestName }}_Other(t *testing.T) {
resultObj, expectedObj user
resultTime, expectedTime time.Time
value int
+ actualFields, expectedFields []string
)
// Invalid.
@@ -503,6 +514,7 @@ func {{ .CheckerName.AsTestName }}_Ignored(t *testing.T) {
resultPtr, expectedPtr *int
value int
expectedTime time.Time
+ expectedFields, interfaces, directions []string
)
{{ range $ai, $assrn := $.IgnoredAssertions }}
diff --git a/internal/testgen/gen_len.go b/internal/testgen/gen_len.go
index 4b3140da..ade360a8 100644
--- a/internal/testgen/gen_len.go
+++ b/internal/testgen/gen_len.go
@@ -31,28 +31,41 @@ func (g LenTestsGenerator) TemplateData() any {
{Fn: "Equal", Argsf: "len(arr), 42", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "Equal", Argsf: "42, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "Equal", Argsf: "value, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, value"},
+ {Fn: "Equal", Argsf: "len(expArr), len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, len(expArr)"},
+
{Fn: "EqualValues", Argsf: "len(arr), 42", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "EqualValues", Argsf: "42, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "EqualValues", Argsf: "value, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, value"},
+ {
+ Fn: "EqualValues", Argsf: "len(expArr), len(arr)", ReportMsgf: report,
+ ProposedFn: proposedFn, ProposedArgsf: "arr, len(expArr)",
+ },
+
{Fn: "Exactly", Argsf: "len(arr), 42", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "Exactly", Argsf: "42, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "Exactly", Argsf: "value, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, value"},
+ {Fn: "Exactly", Argsf: "len(expArr), len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, len(expArr)"},
+
{Fn: "True", Argsf: "len(arr) == 42", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
{Fn: "True", Argsf: "42 == len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, 42"},
+ {Fn: "True", Argsf: "len(arr) == value", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, value"},
+ {Fn: "True", Argsf: "len(arr) == len(expArr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, len(expArr)"},
+
+ // Constant case.
+ {Fn: "Equal", Argsf: "constNum, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, constNum"},
+ {Fn: "EqualValues", Argsf: "constNum, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, constNum"},
+ {Fn: "Exactly", Argsf: "constNum, len(arr)", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, constNum"},
+ {Fn: "True", Argsf: "len(arr) == constNum", ReportMsgf: report, ProposedFn: proposedFn, ProposedArgsf: "arr, constNum"},
},
ValidAssertions: []Assertion{
{Fn: "Len", Argsf: "arr, 42"},
+ {Fn: "Len", Argsf: "arr, value"},
{Fn: "Len", Argsf: "arr, len(arr)"},
},
IgnoredAssertions: []Assertion{
- {Fn: "Equal", Argsf: "len(arr), len(arr)"},
{Fn: "Equal", Argsf: "len(arr), value"},
- {Fn: "EqualValues", Argsf: "len(arr), len(arr)"},
{Fn: "EqualValues", Argsf: "len(arr), value"},
- {Fn: "Exactly", Argsf: "len(arr), len(arr)"},
{Fn: "Exactly", Argsf: "len(arr), value"},
- {Fn: "True", Argsf: "len(arr) == len(arr)"},
- {Fn: "True", Argsf: "len(arr) == value"},
{Fn: "True", Argsf: "value == len(arr)"},
{Fn: "NotEqual", Argsf: "42, len(arr)"},
@@ -117,8 +130,10 @@ import (
"github.com/stretchr/testify/assert"
)
+const constNum = 10
+
func {{ .CheckerName.AsTestName }}(t *testing.T) {
- var arr [3]int
+ var arr, expArr [3]int
var value int
// Invalid.