add timeout tests for Query and QueryRow on db and tx wrappers

This commit is contained in:
Jesse Hallam
2026-05-12 16:38:11 -03:00
parent d19434a51b
commit 2e20cc9e82
@@ -5,10 +5,13 @@ package sqlstore
import (
"context"
"errors"
"strings"
"sync"
"testing"
"time"
"github.com/lib/pq"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
@@ -16,6 +19,21 @@ import (
"github.com/mattermost/mattermost/server/public/shared/mlog"
)
// requireQueryTimeout asserts that err represents a query-timeout cancellation.
// The error is either context.DeadlineExceeded (context fires before the query
// reaches the server) or a pq query_canceled error (PostgreSQL error 57014,
// fired when the driver cancels an in-flight query).
func requireQueryTimeout(t *testing.T, err error) {
t.Helper()
require.Error(t, err)
var pqErr *pq.Error
if errors.As(err, &pqErr) {
require.Equal(t, pq.ErrorCode("57014"), pqErr.Code)
} else {
require.ErrorIs(t, err, context.DeadlineExceeded)
}
}
func TestSqlX(t *testing.T) {
t.Run("NamedParse", func(t *testing.T) {
queries := []struct {
@@ -50,61 +68,129 @@ func TestSqlX(t *testing.T) {
})
}
func TestSqlxSelect(t *testing.T) {
testDrivers := []string{
model.DatabaseDriverPostgres,
func makeStoreWithTimeout(t *testing.T, driver string, queryTimeout time.Duration) *SqlStore {
t.Helper()
settings, err := makeSqlSettings(driver)
if err != nil {
t.Skip(err)
}
for _, driver := range testDrivers {
t.Run(driver, func(t *testing.T) {
settings, err := makeSqlSettings(driver)
if err != nil {
t.Skip(err)
}
*settings.QueryTimeout = 1
store := &SqlStore{
rrCounter: 0,
srCounter: 0,
settings: settings,
logger: mlog.CreateConsoleTestLogger(t),
quitMonitor: make(chan struct{}),
wgMonitor: &sync.WaitGroup{},
}
require.NoError(t, store.initConnection())
defer store.Close()
t.Run("SelectCtx", func(t *testing.T) {
var result []string
err := store.GetMaster().SelectContext(context.Background(), &result, "SELECT 'test' AS col")
require.NoError(t, err)
require.Equal(t, []string{"test"}, result)
// Test timeout
ctx, cancel := context.WithTimeout(context.Background(), 1)
defer cancel()
query := "SELECT pg_sleep(2)"
err = store.GetMaster().SelectContext(ctx, &result, query)
require.Error(t, err)
require.Equal(t, context.DeadlineExceeded, err)
})
t.Run("SelectBuilderCtx", func(t *testing.T) {
var result []string
builder := store.getQueryBuilder().
Select("'test' AS col")
err := store.GetMaster().SelectBuilderCtx(context.Background(), &result, builder)
require.NoError(t, err)
require.Equal(t, []string{"test"}, result)
// Test timeout
ctx, cancel := context.WithTimeout(context.Background(), 1)
defer cancel()
builder = store.getQueryBuilder().
Select("pg_sleep(2)")
err = store.GetMaster().SelectBuilderCtx(ctx, &result, builder)
require.Error(t, err)
require.Equal(t, context.DeadlineExceeded, err)
})
})
*settings.QueryTimeout = int(queryTimeout.Seconds())
store := &SqlStore{
settings: settings,
logger: mlog.CreateConsoleTestLogger(t),
quitMonitor: make(chan struct{}),
wgMonitor: &sync.WaitGroup{},
}
require.NoError(t, store.initConnection())
t.Cleanup(store.Close)
return store
}
func TestSqlxQuery(t *testing.T) {
store := makeStoreWithTimeout(t, model.DatabaseDriverPostgres, 1*time.Second)
t.Run("db/happy", func(t *testing.T) {
rows, err := store.GetMaster().Query("SELECT 1")
require.NoError(t, err)
defer rows.Close()
require.True(t, rows.Next())
var v int
require.NoError(t, rows.Scan(&v))
require.Equal(t, 1, v)
})
t.Run("db/timeout", func(t *testing.T) {
_, err := store.GetMaster().Query("SELECT pg_sleep(2)")
requireQueryTimeout(t, err)
})
t.Run("tx/happy", func(t *testing.T) {
tx, err := store.GetMaster().Begin()
require.NoError(t, err)
defer func() { assert.NoError(t, tx.Rollback()) }()
rows, err := tx.Query("SELECT 1")
require.NoError(t, err)
defer rows.Close()
require.True(t, rows.Next())
var v int
require.NoError(t, rows.Scan(&v))
require.Equal(t, 1, v)
})
t.Run("tx/timeout", func(t *testing.T) {
tx, err := store.GetMaster().Begin()
require.NoError(t, err)
defer func() { assert.NoError(t, tx.Rollback()) }()
_, err = tx.Query("SELECT pg_sleep(2)")
requireQueryTimeout(t, err)
})
}
func TestSqlxQueryRow(t *testing.T) {
store := makeStoreWithTimeout(t, model.DatabaseDriverPostgres, 1*time.Second)
t.Run("db/happy", func(t *testing.T) {
var v int
require.NoError(t, store.GetMaster().QueryRow("SELECT 1").Scan(&v))
require.Equal(t, 1, v)
})
t.Run("db/timeout", func(t *testing.T) {
var v int
err := store.GetMaster().QueryRow("SELECT 1 FROM (SELECT pg_sleep(2)) s").Scan(&v)
requireQueryTimeout(t, err)
})
t.Run("tx/happy", func(t *testing.T) {
tx, err := store.GetMaster().Begin()
require.NoError(t, err)
defer func() { assert.NoError(t, tx.Rollback()) }()
var v int
require.NoError(t, tx.QueryRow("SELECT 1").Scan(&v))
require.Equal(t, 1, v)
})
t.Run("tx/timeout", func(t *testing.T) {
tx, err := store.GetMaster().Begin()
require.NoError(t, err)
defer func() { assert.NoError(t, tx.Rollback()) }()
var v int
err = tx.QueryRow("SELECT 1 FROM (SELECT pg_sleep(2)) s").Scan(&v)
requireQueryTimeout(t, err)
})
}
func TestSqlxSelect(t *testing.T) {
store := makeStoreWithTimeout(t, model.DatabaseDriverPostgres, 1*time.Second)
t.Run("SelectCtx", func(t *testing.T) {
var result []string
err := store.GetMaster().SelectContext(context.Background(), &result, "SELECT 'test' AS col")
require.NoError(t, err)
require.Equal(t, []string{"test"}, result)
// Test timeout
ctx, cancel := context.WithTimeout(context.Background(), 1)
defer cancel()
query := "SELECT pg_sleep(2)"
err = store.GetMaster().SelectContext(ctx, &result, query)
requireQueryTimeout(t, err)
})
t.Run("SelectBuilderCtx", func(t *testing.T) {
var result []string
builder := store.getQueryBuilder().
Select("'test' AS col")
err := store.GetMaster().SelectBuilderCtx(context.Background(), &result, builder)
require.NoError(t, err)
require.Equal(t, []string{"test"}, result)
// Test timeout
ctx, cancel := context.WithTimeout(context.Background(), 1)
defer cancel()
builder = store.getQueryBuilder().
Select("pg_sleep(2)")
err = store.GetMaster().SelectBuilderCtx(ctx, &result, builder)
requireQueryTimeout(t, err)
})
}