mirror of https://github.com/VinGarcia/ksql.git
Add tests for the error messages produced by the Mock()
parent
6f2ecbef5a
commit
0ff7a92d72
|
@ -0,0 +1,70 @@
|
|||
package tt
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
)
|
||||
|
||||
// AssertEqual will compare the got argument with the expected argument
|
||||
// and fail the test with an appropriate error message if they don't match.
|
||||
func AssertEqual(t *testing.T, got interface{}, expected interface{}, msg ...interface{}) {
|
||||
require.Equal(t, expected, got, msg...)
|
||||
}
|
||||
|
||||
// AssertNotEqual will compare the got argument with the expected argument
|
||||
// and fail the test with an appropriate error message if they match.
|
||||
func AssertNotEqual(t *testing.T, got interface{}, expected interface{}, msg ...interface{}) {
|
||||
require.NotEqual(t, expected, got, msg...)
|
||||
}
|
||||
|
||||
// AssertNoErr will check if the input error is nil, and if not
|
||||
// it will fail the test with an appropriate error message.
|
||||
func AssertNoErr(t *testing.T, err error) {
|
||||
require.Equal(t, nil, err, "received unexpected error: %s", err)
|
||||
}
|
||||
|
||||
// AssertErrContains will first check if the error that the error
|
||||
// indeed is not nil, and then check if its error message contains
|
||||
// all the substrs specified on the substrs argument.
|
||||
//
|
||||
// In case either assertion fails it will fail the test with
|
||||
// an appropriate error message.
|
||||
func AssertErrContains(t *testing.T, err error, substrs ...string) {
|
||||
require.NotEqual(t, nil, err, "expected an error but the error is nil")
|
||||
|
||||
msg := err.Error()
|
||||
|
||||
for _, substr := range substrs {
|
||||
require.True(t,
|
||||
strings.Contains(msg, substr),
|
||||
"missing substring '%s' in error message: '%s'",
|
||||
substr, msg,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
// AssertApproxDuration checks if the durations v1 and v2 are close up to the tolerance specified.
|
||||
// The format and args slice can be used for generating an appropriate error message if they are not.
|
||||
func AssertApproxDuration(t *testing.T, tolerance time.Duration, v1, v2 time.Duration, format string, args ...interface{}) {
|
||||
diff := v1 - v2
|
||||
if diff < 0 {
|
||||
diff = -diff
|
||||
}
|
||||
|
||||
require.True(t, diff <= tolerance, fmt.Sprintf(format, args...))
|
||||
}
|
||||
|
||||
// AssertApproxTime checks if the times v1 and v2 are close up to the tolerance specified.
|
||||
// The format and args slice can be used for generating an appropriate error message if they are not.
|
||||
func AssertApproxTime(t *testing.T, tolerance time.Duration, v1, v2 time.Time, format string, args ...interface{}) {
|
||||
diff := v1.Sub(v2)
|
||||
if diff < 0 {
|
||||
diff = -diff
|
||||
}
|
||||
|
||||
require.True(t, diff <= tolerance, fmt.Sprintf(format, args...))
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
package tt
|
||||
|
||||
// PanicHandler will run the input function and recover
|
||||
// from any panics it might generate.
|
||||
//
|
||||
// It will then save the panic payload and return it
|
||||
// so it can be asserted by other functions on the test.
|
||||
func PanicHandler(fn func()) (panicPayload interface{}) {
|
||||
defer func() {
|
||||
// Overwrites the panic payload if a pannic actually occurs:
|
||||
if r := recover(); r != nil {
|
||||
panicPayload = r
|
||||
}
|
||||
}()
|
||||
|
||||
fn()
|
||||
return nil
|
||||
}
|
14
mocks.go
14
mocks.go
|
@ -81,7 +81,7 @@ func (m Mock) SetFallbackDatabase(db Provider) Mock {
|
|||
// Insert ...
|
||||
func (m Mock) Insert(ctx context.Context, table Table, record interface{}) error {
|
||||
if m.InsertFn == nil {
|
||||
panic(fmt.Errorf("Mock.Insert(ctx, %v, %v) called but the ksql.Mock.InsertFn() is not set", table, record))
|
||||
panic(fmt.Errorf("ksql.Mock.Insert(ctx, %v, %v) called but the ksql.Mock.InsertFn() is not set", table, record))
|
||||
}
|
||||
return m.InsertFn(ctx, table, record)
|
||||
}
|
||||
|
@ -89,7 +89,7 @@ func (m Mock) Insert(ctx context.Context, table Table, record interface{}) error
|
|||
// Update ...
|
||||
func (m Mock) Update(ctx context.Context, table Table, record interface{}) error {
|
||||
if m.UpdateFn == nil {
|
||||
panic(fmt.Errorf("Mock.Update(ctx, %v, %v) called but the ksql.Mock.UpdateFn() is not set", table, record))
|
||||
panic(fmt.Errorf("ksql.Mock.Update(ctx, %v, %v) called but the ksql.Mock.UpdateFn() is not set", table, record))
|
||||
}
|
||||
return m.UpdateFn(ctx, table, record)
|
||||
}
|
||||
|
@ -97,7 +97,7 @@ func (m Mock) Update(ctx context.Context, table Table, record interface{}) error
|
|||
// Delete ...
|
||||
func (m Mock) Delete(ctx context.Context, table Table, idOrRecord interface{}) error {
|
||||
if m.DeleteFn == nil {
|
||||
panic(fmt.Errorf("Mock.Delete(ctx, %v, %v) called but the ksql.Mock.DeleteFn() is not set", table, idOrRecord))
|
||||
panic(fmt.Errorf("ksql.Mock.Delete(ctx, %v, %v) called but the ksql.Mock.DeleteFn() is not set", table, idOrRecord))
|
||||
}
|
||||
return m.DeleteFn(ctx, table, idOrRecord)
|
||||
}
|
||||
|
@ -105,7 +105,7 @@ func (m Mock) Delete(ctx context.Context, table Table, idOrRecord interface{}) e
|
|||
// Query ...
|
||||
func (m Mock) Query(ctx context.Context, records interface{}, query string, params ...interface{}) error {
|
||||
if m.QueryFn == nil {
|
||||
panic(fmt.Errorf("Mock.Query(ctx, %v, %s, %v) called but the ksql.Mock.QueryFn() is not set", records, query, params))
|
||||
panic(fmt.Errorf("ksql.Mock.Query(ctx, %v, %s, %v) called but the ksql.Mock.QueryFn() is not set", records, query, params))
|
||||
}
|
||||
return m.QueryFn(ctx, records, query, params...)
|
||||
}
|
||||
|
@ -113,7 +113,7 @@ func (m Mock) Query(ctx context.Context, records interface{}, query string, para
|
|||
// QueryOne ...
|
||||
func (m Mock) QueryOne(ctx context.Context, record interface{}, query string, params ...interface{}) error {
|
||||
if m.QueryOneFn == nil {
|
||||
panic(fmt.Errorf("Mock.QueryOne(ctx, %v, %s, %v) called but the ksql.Mock.QueryOneFn() is not set", record, query, params))
|
||||
panic(fmt.Errorf("ksql.Mock.QueryOne(ctx, %v, %s, %v) called but the ksql.Mock.QueryOneFn() is not set", record, query, params))
|
||||
}
|
||||
return m.QueryOneFn(ctx, record, query, params...)
|
||||
}
|
||||
|
@ -121,7 +121,7 @@ func (m Mock) QueryOne(ctx context.Context, record interface{}, query string, pa
|
|||
// QueryChunks ...
|
||||
func (m Mock) QueryChunks(ctx context.Context, parser ChunkParser) error {
|
||||
if m.QueryChunksFn == nil {
|
||||
panic(fmt.Errorf("Mock.QueryChunks(ctx, %v) called but the ksql.Mock.QueryChunksFn() is not set", parser))
|
||||
panic(fmt.Errorf("ksql.Mock.QueryChunks(ctx, %v) called but the ksql.Mock.QueryChunksFn() is not set", parser))
|
||||
}
|
||||
return m.QueryChunksFn(ctx, parser)
|
||||
}
|
||||
|
@ -129,7 +129,7 @@ func (m Mock) QueryChunks(ctx context.Context, parser ChunkParser) error {
|
|||
// Exec ...
|
||||
func (m Mock) Exec(ctx context.Context, query string, params ...interface{}) (rowsAffected int64, _ error) {
|
||||
if m.ExecFn == nil {
|
||||
panic(fmt.Errorf("Mock.Exec(ctx, %s, %v) called but the ksql.Mock.ExecFn() is not set", query, params))
|
||||
panic(fmt.Errorf("ksql.Mock.Exec(ctx, %s, %v) called but the ksql.Mock.ExecFn() is not set", query, params))
|
||||
}
|
||||
return m.ExecFn(ctx, query, params...)
|
||||
}
|
||||
|
|
|
@ -0,0 +1,145 @@
|
|||
// This test was written mostly for test coverage since the mock is trivial
|
||||
package ksql_test
|
||||
|
||||
import (
|
||||
"context"
|
||||
"testing"
|
||||
|
||||
"github.com/vingarcia/ksql"
|
||||
tt "github.com/vingarcia/ksql/internal/testtools"
|
||||
)
|
||||
|
||||
func TestMock(t *testing.T) {
|
||||
UsersTable := ksql.NewTable("users", "id")
|
||||
type User struct {
|
||||
ID int `ksql:"id"`
|
||||
Name string `ksql:"name"`
|
||||
Age int `ksql:"age"`
|
||||
}
|
||||
|
||||
t.Run("testing unset behaviors for all methods", func(t *testing.T) {
|
||||
t.Run("Insert should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
mock.Insert(ctx, UsersTable, &User{
|
||||
Name: "fake-name",
|
||||
Age: 42,
|
||||
})
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.Insert(", "ksql.Mock.InsertFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("Update should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
mock.Update(ctx, UsersTable, &User{
|
||||
ID: 4242,
|
||||
Name: "fake-name",
|
||||
Age: 42,
|
||||
})
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.Update(", "ksql.Mock.UpdateFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("Delete should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
mock.Delete(ctx, UsersTable, &User{
|
||||
ID: 4242,
|
||||
Name: "fake-name",
|
||||
Age: 42,
|
||||
})
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.Delete(", "ksql.Mock.DeleteFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("Query should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
var users []User
|
||||
mock.Query(ctx, &users, "SELECT * FROM user WHERE age = ?", 42)
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.Query(", "ksql.Mock.QueryFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("QueryOne should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
var user User
|
||||
mock.QueryOne(ctx, &user, "SELECT * FROM user WHERE id = ?", 4242)
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.QueryOne(", "ksql.Mock.QueryOneFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("QueryChunks should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
var users []User
|
||||
mock.QueryChunks(ctx, ksql.ChunkParser{
|
||||
Query: "SELECT * FROM users WHERE age = ?",
|
||||
Params: []interface{}{
|
||||
4242,
|
||||
},
|
||||
ChunkSize: 10,
|
||||
ForEachChunk: func(chunk []User) error {
|
||||
users = append(users, chunk...)
|
||||
return nil
|
||||
},
|
||||
})
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.QueryChunks(", "ksql.Mock.QueryChunksFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("Exec should panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
mock.Exec(ctx, "INSERT INTO users_permissions(user_id, permission_id) VALUES (?, ?)", 4242, 4)
|
||||
})
|
||||
|
||||
err, ok := panicPayload.(error)
|
||||
tt.AssertEqual(t, ok, true)
|
||||
tt.AssertErrContains(t, err, "ksql.Mock.Exec(", "ksql.Mock.ExecFn", "not set")
|
||||
})
|
||||
|
||||
t.Run("Transaction should not panic", func(t *testing.T) {
|
||||
ctx := context.Background()
|
||||
mock := ksql.Mock{}
|
||||
|
||||
executed := false
|
||||
panicPayload := tt.PanicHandler(func() {
|
||||
mock.Transaction(ctx, func(db ksql.Provider) error {
|
||||
executed = true
|
||||
return nil
|
||||
})
|
||||
})
|
||||
|
||||
tt.AssertEqual(t, panicPayload, nil)
|
||||
tt.AssertEqual(t, executed, true)
|
||||
})
|
||||
})
|
||||
}
|
Loading…
Reference in New Issue