mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
834 lines
21 KiB
Go
834 lines
21 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package sqlstore
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"regexp"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/go-sql-driver/mysql"
|
|
"github.com/lib/pq"
|
|
"github.com/mattermost/gorp"
|
|
"github.com/pkg/errors"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/mattermost/mattermost-server/v5/einterfaces/mocks"
|
|
"github.com/mattermost/mattermost-server/v5/model"
|
|
"github.com/mattermost/mattermost-server/v5/store"
|
|
"github.com/mattermost/mattermost-server/v5/store/searchtest"
|
|
"github.com/mattermost/mattermost-server/v5/store/storetest"
|
|
)
|
|
|
|
type storeType struct {
|
|
Name string
|
|
SqlSettings *model.SqlSettings
|
|
SqlStore *SqlStore
|
|
Store store.Store
|
|
}
|
|
|
|
var storeTypes []*storeType
|
|
|
|
func newStoreType(name, driver string) *storeType {
|
|
return &storeType{
|
|
Name: name,
|
|
SqlSettings: storetest.MakeSqlSettings(driver, false),
|
|
}
|
|
}
|
|
|
|
func StoreTest(t *testing.T, f func(*testing.T, store.Store)) {
|
|
defer func() {
|
|
if err := recover(); err != nil {
|
|
tearDownStores()
|
|
panic(err)
|
|
}
|
|
}()
|
|
for _, st := range storeTypes {
|
|
st := st
|
|
t.Run(st.Name, func(t *testing.T) {
|
|
if testing.Short() {
|
|
t.SkipNow()
|
|
}
|
|
f(t, st.Store)
|
|
})
|
|
}
|
|
}
|
|
|
|
func StoreTestWithSearchTestEngine(t *testing.T, f func(*testing.T, store.Store, *searchtest.SearchTestEngine)) {
|
|
defer func() {
|
|
if err := recover(); err != nil {
|
|
tearDownStores()
|
|
panic(err)
|
|
}
|
|
}()
|
|
|
|
for _, st := range storeTypes {
|
|
st := st
|
|
searchTestEngine := &searchtest.SearchTestEngine{
|
|
Driver: *st.SqlSettings.DriverName,
|
|
}
|
|
|
|
t.Run(st.Name, func(t *testing.T) { f(t, st.Store, searchTestEngine) })
|
|
}
|
|
}
|
|
|
|
func StoreTestWithSqlStore(t *testing.T, f func(*testing.T, store.Store, storetest.SqlStore)) {
|
|
defer func() {
|
|
if err := recover(); err != nil {
|
|
tearDownStores()
|
|
panic(err)
|
|
}
|
|
}()
|
|
for _, st := range storeTypes {
|
|
st := st
|
|
t.Run(st.Name, func(t *testing.T) {
|
|
if testing.Short() {
|
|
t.SkipNow()
|
|
}
|
|
f(t, st.Store, st.SqlStore)
|
|
})
|
|
}
|
|
}
|
|
|
|
func initStores() {
|
|
if testing.Short() {
|
|
return
|
|
}
|
|
// In CI, we already run the entire test suite for both mysql and postgres in parallel.
|
|
// So we just run the tests for the current database set.
|
|
if os.Getenv("IS_CI") == "true" {
|
|
switch os.Getenv("MM_SQLSETTINGS_DRIVERNAME") {
|
|
case "mysql":
|
|
storeTypes = append(storeTypes, newStoreType("MySQL", model.DatabaseDriverMysql))
|
|
case "postgres":
|
|
storeTypes = append(storeTypes, newStoreType("PostgreSQL", model.DatabaseDriverPostgres))
|
|
}
|
|
} else {
|
|
storeTypes = append(storeTypes,
|
|
newStoreType("MySQL", model.DatabaseDriverMysql),
|
|
newStoreType("PostgreSQL", model.DatabaseDriverPostgres),
|
|
)
|
|
}
|
|
|
|
defer func() {
|
|
if err := recover(); err != nil {
|
|
tearDownStores()
|
|
panic(err)
|
|
}
|
|
}()
|
|
var wg sync.WaitGroup
|
|
for _, st := range storeTypes {
|
|
st := st
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
st.SqlStore = New(*st.SqlSettings, nil)
|
|
st.Store = st.SqlStore
|
|
st.Store.DropAllTables()
|
|
st.Store.MarkSystemRanUnitTests()
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
}
|
|
|
|
var tearDownStoresOnce sync.Once
|
|
|
|
func tearDownStores() {
|
|
if testing.Short() {
|
|
return
|
|
}
|
|
tearDownStoresOnce.Do(func() {
|
|
var wg sync.WaitGroup
|
|
wg.Add(len(storeTypes))
|
|
for _, st := range storeTypes {
|
|
st := st
|
|
go func() {
|
|
if st.Store != nil {
|
|
st.Store.Close()
|
|
}
|
|
if st.SqlSettings != nil {
|
|
storetest.CleanupSqlSettings(st.SqlSettings)
|
|
}
|
|
wg.Done()
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
})
|
|
}
|
|
|
|
// This test was used to consistently reproduce the race
|
|
// before the fix in MM-28397.
|
|
// Keeping it here to help avoiding future regressions.
|
|
func TestStoreLicenseRace(t *testing.T) {
|
|
settings := makeSqlSettings(model.DatabaseDriverPostgres)
|
|
store := New(*settings, nil)
|
|
defer func() {
|
|
store.Close()
|
|
storetest.CleanupSqlSettings(settings)
|
|
}()
|
|
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(3)
|
|
|
|
go func() {
|
|
store.UpdateLicense(&model.License{})
|
|
wg.Done()
|
|
}()
|
|
|
|
go func() {
|
|
store.GetReplica()
|
|
wg.Done()
|
|
}()
|
|
|
|
go func() {
|
|
store.GetSearchReplica()
|
|
wg.Done()
|
|
}()
|
|
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestGetReplica(t *testing.T) {
|
|
t.Parallel()
|
|
testCases := []struct {
|
|
Description string
|
|
DataSourceReplicaNum int
|
|
DataSourceSearchReplicaNum int
|
|
}{
|
|
{
|
|
"no replicas",
|
|
0,
|
|
0,
|
|
},
|
|
{
|
|
"one source replica",
|
|
1,
|
|
0,
|
|
},
|
|
{
|
|
"multiple source replicas",
|
|
3,
|
|
0,
|
|
},
|
|
{
|
|
"one source search replica",
|
|
0,
|
|
1,
|
|
},
|
|
{
|
|
"multiple source search replicas",
|
|
0,
|
|
3,
|
|
},
|
|
{
|
|
"one source replica, one source search replica",
|
|
1,
|
|
1,
|
|
},
|
|
{
|
|
"one source replica, multiple source search replicas",
|
|
1,
|
|
3,
|
|
},
|
|
{
|
|
"multiple source replica, one source search replica",
|
|
3,
|
|
1,
|
|
},
|
|
{
|
|
"multiple source replica, multiple source search replicas",
|
|
3,
|
|
3,
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
testCase := testCase
|
|
t.Run(testCase.Description+" with license", func(t *testing.T) {
|
|
|
|
settings := makeSqlSettings(model.DatabaseDriverPostgres)
|
|
dataSourceReplicas := []string{}
|
|
dataSourceSearchReplicas := []string{}
|
|
for i := 0; i < testCase.DataSourceReplicaNum; i++ {
|
|
dataSourceReplicas = append(dataSourceReplicas, *settings.DataSource)
|
|
}
|
|
for i := 0; i < testCase.DataSourceSearchReplicaNum; i++ {
|
|
dataSourceSearchReplicas = append(dataSourceSearchReplicas, *settings.DataSource)
|
|
}
|
|
|
|
settings.DataSourceReplicas = dataSourceReplicas
|
|
settings.DataSourceSearchReplicas = dataSourceSearchReplicas
|
|
store := New(*settings, nil)
|
|
defer func() {
|
|
store.Close()
|
|
storetest.CleanupSqlSettings(settings)
|
|
}()
|
|
|
|
store.UpdateLicense(&model.License{})
|
|
|
|
replicas := make(map[*gorp.DbMap]bool)
|
|
for i := 0; i < 5; i++ {
|
|
replicas[store.GetReplica()] = true
|
|
}
|
|
|
|
searchReplicas := make(map[*gorp.DbMap]bool)
|
|
for i := 0; i < 5; i++ {
|
|
searchReplicas[store.GetSearchReplica()] = true
|
|
}
|
|
|
|
if testCase.DataSourceReplicaNum > 0 {
|
|
// If replicas were defined, ensure none are the master.
|
|
assert.Len(t, replicas, testCase.DataSourceReplicaNum)
|
|
|
|
for replica := range replicas {
|
|
assert.NotSame(t, store.GetMaster(), replica)
|
|
}
|
|
|
|
} else if assert.Len(t, replicas, 1) {
|
|
// Otherwise ensure the replicas contains only the master.
|
|
for replica := range replicas {
|
|
assert.Same(t, store.GetMaster(), replica)
|
|
}
|
|
}
|
|
|
|
if testCase.DataSourceSearchReplicaNum > 0 {
|
|
// If search replicas were defined, ensure none are the master nor the replicas.
|
|
assert.Len(t, searchReplicas, testCase.DataSourceSearchReplicaNum)
|
|
|
|
for searchReplica := range searchReplicas {
|
|
assert.NotSame(t, store.GetMaster(), searchReplica)
|
|
for replica := range replicas {
|
|
assert.NotSame(t, searchReplica, replica)
|
|
}
|
|
}
|
|
} else if testCase.DataSourceReplicaNum > 0 {
|
|
assert.Equal(t, len(replicas), len(searchReplicas))
|
|
for k := range replicas {
|
|
assert.True(t, searchReplicas[k])
|
|
}
|
|
} else if testCase.DataSourceReplicaNum == 0 && assert.Len(t, searchReplicas, 1) {
|
|
// Otherwise ensure the search replicas contains the master.
|
|
for searchReplica := range searchReplicas {
|
|
assert.Same(t, store.GetMaster(), searchReplica)
|
|
}
|
|
}
|
|
})
|
|
|
|
t.Run(testCase.Description+" without license", func(t *testing.T) {
|
|
|
|
settings := makeSqlSettings(model.DatabaseDriverPostgres)
|
|
dataSourceReplicas := []string{}
|
|
dataSourceSearchReplicas := []string{}
|
|
for i := 0; i < testCase.DataSourceReplicaNum; i++ {
|
|
dataSourceReplicas = append(dataSourceReplicas, *settings.DataSource)
|
|
}
|
|
for i := 0; i < testCase.DataSourceSearchReplicaNum; i++ {
|
|
dataSourceSearchReplicas = append(dataSourceSearchReplicas, *settings.DataSource)
|
|
}
|
|
|
|
settings.DataSourceReplicas = dataSourceReplicas
|
|
settings.DataSourceSearchReplicas = dataSourceSearchReplicas
|
|
store := New(*settings, nil)
|
|
defer func() {
|
|
store.Close()
|
|
storetest.CleanupSqlSettings(settings)
|
|
}()
|
|
|
|
replicas := make(map[*gorp.DbMap]bool)
|
|
for i := 0; i < 5; i++ {
|
|
replicas[store.GetReplica()] = true
|
|
}
|
|
|
|
searchReplicas := make(map[*gorp.DbMap]bool)
|
|
for i := 0; i < 5; i++ {
|
|
searchReplicas[store.GetSearchReplica()] = true
|
|
}
|
|
|
|
if testCase.DataSourceReplicaNum > 0 {
|
|
// If replicas were defined, ensure none are the master.
|
|
assert.Len(t, replicas, 1)
|
|
|
|
for replica := range replicas {
|
|
assert.Same(t, store.GetMaster(), replica)
|
|
}
|
|
|
|
} else if assert.Len(t, replicas, 1) {
|
|
// Otherwise ensure the replicas contains only the master.
|
|
for replica := range replicas {
|
|
assert.Same(t, store.GetMaster(), replica)
|
|
}
|
|
}
|
|
|
|
if testCase.DataSourceSearchReplicaNum > 0 {
|
|
// If search replicas were defined, ensure none are the master nor the replicas.
|
|
assert.Len(t, searchReplicas, 1)
|
|
|
|
for searchReplica := range searchReplicas {
|
|
assert.Same(t, store.GetMaster(), searchReplica)
|
|
}
|
|
|
|
} else if testCase.DataSourceReplicaNum > 0 {
|
|
assert.Equal(t, len(replicas), len(searchReplicas))
|
|
for k := range replicas {
|
|
assert.True(t, searchReplicas[k])
|
|
}
|
|
} else if assert.Len(t, searchReplicas, 1) {
|
|
// Otherwise ensure the search replicas contains the master.
|
|
for searchReplica := range searchReplicas {
|
|
assert.Same(t, store.GetMaster(), searchReplica)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGetDbVersion(t *testing.T) {
|
|
testDrivers := []string{
|
|
model.DatabaseDriverPostgres,
|
|
model.DatabaseDriverMysql,
|
|
}
|
|
|
|
for _, driver := range testDrivers {
|
|
t.Run("Should return db version for "+driver, func(t *testing.T) {
|
|
t.Parallel()
|
|
settings := makeSqlSettings(driver)
|
|
store := New(*settings, nil)
|
|
|
|
version, err := store.GetDbVersion(false)
|
|
require.NoError(t, err)
|
|
require.Regexp(t, regexp.MustCompile(`\d+\.\d+(\.\d+)?`), version)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestUpAndDownMigrations(t *testing.T) {
|
|
testDrivers := []string{
|
|
model.DatabaseDriverPostgres,
|
|
model.DatabaseDriverMysql,
|
|
}
|
|
|
|
for _, driver := range testDrivers {
|
|
t.Run("Should be reversible for "+driver, func(t *testing.T) {
|
|
t.Parallel()
|
|
settings := makeSqlSettings(driver)
|
|
store := New(*settings, nil)
|
|
defer store.Close()
|
|
|
|
err := store.migrate(migrationsDirectionDown)
|
|
assert.NoError(t, err, "downing migrations should not error")
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGetAllConns(t *testing.T) {
|
|
t.Parallel()
|
|
testCases := []struct {
|
|
Description string
|
|
DataSourceReplicaNum int
|
|
DataSourceSearchReplicaNum int
|
|
ExpectedNumConnections int
|
|
}{
|
|
{
|
|
"no replicas",
|
|
0,
|
|
0,
|
|
1,
|
|
},
|
|
{
|
|
"one source replica",
|
|
1,
|
|
0,
|
|
2,
|
|
},
|
|
{
|
|
"multiple source replicas",
|
|
3,
|
|
0,
|
|
4,
|
|
},
|
|
{
|
|
"one source search replica",
|
|
0,
|
|
1,
|
|
1,
|
|
},
|
|
{
|
|
"multiple source search replicas",
|
|
0,
|
|
3,
|
|
1,
|
|
},
|
|
{
|
|
"one source replica, one source search replica",
|
|
1,
|
|
1,
|
|
2,
|
|
},
|
|
{
|
|
"one source replica, multiple source search replicas",
|
|
1,
|
|
3,
|
|
2,
|
|
},
|
|
{
|
|
"multiple source replica, one source search replica",
|
|
3,
|
|
1,
|
|
4,
|
|
},
|
|
{
|
|
"multiple source replica, multiple source search replicas",
|
|
3,
|
|
3,
|
|
4,
|
|
},
|
|
}
|
|
|
|
for _, testCase := range testCases {
|
|
testCase := testCase
|
|
t.Run(testCase.Description, func(t *testing.T) {
|
|
t.Parallel()
|
|
settings := makeSqlSettings(model.DatabaseDriverPostgres)
|
|
dataSourceReplicas := []string{}
|
|
dataSourceSearchReplicas := []string{}
|
|
for i := 0; i < testCase.DataSourceReplicaNum; i++ {
|
|
dataSourceReplicas = append(dataSourceReplicas, *settings.DataSource)
|
|
}
|
|
for i := 0; i < testCase.DataSourceSearchReplicaNum; i++ {
|
|
dataSourceSearchReplicas = append(dataSourceSearchReplicas, *settings.DataSource)
|
|
}
|
|
|
|
settings.DataSourceReplicas = dataSourceReplicas
|
|
settings.DataSourceSearchReplicas = dataSourceSearchReplicas
|
|
store := New(*settings, nil)
|
|
defer func() {
|
|
store.Close()
|
|
storetest.CleanupSqlSettings(settings)
|
|
}()
|
|
|
|
assert.Len(t, store.GetAllConns(), testCase.ExpectedNumConnections)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsDuplicate(t *testing.T) {
|
|
testErrors := map[error]bool{
|
|
&pq.Error{Code: "42P06"}: false,
|
|
&pq.Error{Code: PGDupTableErrorCode}: true,
|
|
&mysql.MySQLError{Number: uint16(1000)}: false,
|
|
&mysql.MySQLError{Number: MySQLDupTableErrorCode}: true,
|
|
errors.New("Random error"): false,
|
|
}
|
|
|
|
for err, expected := range testErrors {
|
|
t.Run(fmt.Sprintf("Should return %t for %s", expected, err.Error()), func(t *testing.T) {
|
|
t.Parallel()
|
|
assert.Equal(t, expected, IsDuplicate(err))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestVersionString(t *testing.T) {
|
|
versions := []struct {
|
|
input int
|
|
output string
|
|
}{
|
|
{
|
|
input: 100000,
|
|
output: "10.0",
|
|
},
|
|
{
|
|
input: 90603,
|
|
output: "9.603",
|
|
},
|
|
{
|
|
input: 120005,
|
|
output: "12.5",
|
|
},
|
|
}
|
|
|
|
for _, v := range versions {
|
|
out := VersionString(v.input)
|
|
assert.Equal(t, v.output, out)
|
|
}
|
|
}
|
|
|
|
func TestReplicaLagQuery(t *testing.T) {
|
|
testDrivers := []string{
|
|
model.DatabaseDriverPostgres,
|
|
model.DatabaseDriverMysql,
|
|
}
|
|
|
|
for _, driver := range testDrivers {
|
|
settings := makeSqlSettings(driver)
|
|
var query string
|
|
var tableName string
|
|
// Just any random query which returns a row in (string, int) format.
|
|
switch driver {
|
|
case model.DatabaseDriverPostgres:
|
|
query = `SELECT relname, count(relname) FROM pg_class WHERE relname='posts' GROUP BY relname`
|
|
tableName = "posts"
|
|
case model.DatabaseDriverMysql:
|
|
query = `SELECT table_name, count(table_name) FROM information_schema.tables WHERE table_name='Posts' and table_schema=Database() GROUP BY table_name`
|
|
tableName = "Posts"
|
|
}
|
|
|
|
settings.ReplicaLagSettings = []*model.ReplicaLagSettings{{
|
|
DataSource: model.NewString(*settings.DataSource),
|
|
QueryAbsoluteLag: model.NewString(query),
|
|
QueryTimeLag: model.NewString(query),
|
|
}}
|
|
|
|
mockMetrics := &mocks.MetricsInterface{}
|
|
defer mockMetrics.AssertExpectations(t)
|
|
mockMetrics.On("SetReplicaLagAbsolute", tableName, float64(1))
|
|
mockMetrics.On("SetReplicaLagTime", tableName, float64(1))
|
|
|
|
store := &SqlStore{
|
|
rrCounter: 0,
|
|
srCounter: 0,
|
|
settings: settings,
|
|
metrics: mockMetrics,
|
|
}
|
|
|
|
store.initConnection()
|
|
store.stores.post = newSqlPostStore(store, mockMetrics)
|
|
err := store.GetMaster().CreateTablesIfNotExists()
|
|
require.NoError(t, err)
|
|
|
|
defer store.Close()
|
|
|
|
err = store.ReplicaLagAbs()
|
|
require.NoError(t, err)
|
|
err = store.ReplicaLagTime()
|
|
require.NoError(t, err)
|
|
}
|
|
}
|
|
|
|
func TestAppendMultipleStatementsFlagMysql(t *testing.T) {
|
|
testCases := []struct {
|
|
Scenario string
|
|
DSN string
|
|
ExpectedDSN string
|
|
Driver string
|
|
}{
|
|
{
|
|
"Should append multiStatements param to the DSN path with existing params",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost?writeTimeout=30s",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost?writeTimeout=30s&multiStatements=true",
|
|
model.DatabaseDriverMysql,
|
|
},
|
|
{
|
|
"Should append multiStatements param to the DSN path with no existing params",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost?multiStatements=true",
|
|
model.DatabaseDriverMysql,
|
|
},
|
|
{
|
|
"Should not multiStatements param to the DSN when driver is not MySQL",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost",
|
|
"user:rand?&ompasswith@character@unix(/var/run/mysqld/mysqld.sock)/mattermost",
|
|
model.DatabaseDriverPostgres,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.Scenario, func(t *testing.T) {
|
|
store := &SqlStore{settings: &model.SqlSettings{DriverName: &tc.Driver, DataSource: &tc.DSN}}
|
|
res, err := store.appendMultipleStatementsFlag(*store.settings.DataSource)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, tc.ExpectedDSN, res)
|
|
})
|
|
}
|
|
}
|
|
|
|
func makeSqlSettings(driver string) *model.SqlSettings {
|
|
switch driver {
|
|
case model.DatabaseDriverPostgres:
|
|
return storetest.MakeSqlSettings(driver, false)
|
|
case model.DatabaseDriverMysql:
|
|
return storetest.MakeSqlSettings(driver, false)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func TestExecNoTimeout(t *testing.T) {
|
|
StoreTest(t, func(t *testing.T, ss store.Store) {
|
|
sqlStore := ss.(*SqlStore)
|
|
var query string
|
|
timeout := sqlStore.master.QueryTimeout
|
|
sqlStore.master.QueryTimeout = 1
|
|
defer func() {
|
|
sqlStore.master.QueryTimeout = timeout
|
|
}()
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
query = `SELECT SLEEP(2);`
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
query = `SELECT pg_sleep(2);`
|
|
}
|
|
_, err := sqlStore.GetMaster().ExecNoTimeout(query)
|
|
require.NoError(t, err)
|
|
})
|
|
}
|
|
|
|
func TestMySQLReadTimeout(t *testing.T) {
|
|
settings := makeSqlSettings(model.DatabaseDriverMysql)
|
|
dataSource := *settings.DataSource
|
|
config, err := mysql.ParseDSN(dataSource)
|
|
require.NoError(t, err)
|
|
|
|
config.ReadTimeout = 1 * time.Second
|
|
dataSource = config.FormatDSN()
|
|
settings.DataSource = &dataSource
|
|
|
|
store := &SqlStore{
|
|
settings: settings,
|
|
}
|
|
store.initConnection()
|
|
defer store.Close()
|
|
|
|
_, err = store.GetMaster().ExecNoTimeout(`SELECT SLEEP(3)`)
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func TestRemoveIndexIfExists(t *testing.T) {
|
|
StoreTest(t, func(t *testing.T, ss store.Store) {
|
|
sqlStore := ss.(*SqlStore)
|
|
|
|
_, err := sqlStore.GetMaster().ExecNoTimeout(`CREATE INDEX idx_posts_create_at ON Posts (CreateAt)`)
|
|
require.Error(t, err)
|
|
|
|
ok := sqlStore.RemoveIndexIfExists("idx_posts_create_at", "Posts")
|
|
require.True(t, ok)
|
|
|
|
ok = sqlStore.RemoveIndexIfExists("idx_posts_create_at", "Posts")
|
|
require.False(t, ok)
|
|
|
|
_, err = sqlStore.GetMaster().ExecNoTimeout(`CREATE INDEX idx_posts_create_at ON Posts (CreateAt)`)
|
|
require.NoError(t, err)
|
|
|
|
ok = sqlStore.RemoveIndexIfExists("idx_posts_create_at", "Posts")
|
|
require.True(t, ok)
|
|
|
|
ok = sqlStore.RemoveIndexIfExists("idx_posts_create_at", "Posts")
|
|
require.False(t, ok)
|
|
})
|
|
}
|
|
|
|
func TestAlterDefaultIfColumnExists(t *testing.T) {
|
|
StoreTest(t, func(t *testing.T, ss store.Store) {
|
|
var query string
|
|
def := new(string)
|
|
sqlStore := ss.(*SqlStore)
|
|
|
|
t.Run("non existent table", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("NotExistent", "NotExistent", nil, nil)
|
|
require.False(t, ok)
|
|
})
|
|
|
|
t.Run("non existent column", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("Posts", "NotExistent", nil, nil)
|
|
require.False(t, ok)
|
|
})
|
|
|
|
t.Run("empty string", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("Posts", "Id", model.NewString(""), model.NewString(""))
|
|
require.True(t, ok)
|
|
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
query = `SELECT column_default
|
|
FROM information_schema.columns
|
|
WHERE table_schema = DATABASE()
|
|
AND table_name = 'Posts'
|
|
AND column_name = 'Id'`
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
query = `SELECT column_default
|
|
FROM information_schema.columns
|
|
WHERE table_name = 'posts'
|
|
AND column_name = 'id'`
|
|
}
|
|
|
|
err := sqlStore.GetMaster().SelectOne(&def, query)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, def)
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
require.Equal(t, "", *def)
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
require.Equal(t, "''::character varying", *def)
|
|
}
|
|
})
|
|
|
|
t.Run("nil input", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("Posts", "Id", nil, nil)
|
|
require.True(t, ok)
|
|
|
|
err := sqlStore.GetMaster().SelectOne(&def, query)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, def)
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
require.Equal(t, "", *def)
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
require.Equal(t, "''::character varying", *def)
|
|
}
|
|
})
|
|
|
|
t.Run("remove", func(t *testing.T) {
|
|
ok := sqlStore.RemoveDefaultIfColumnExists("Posts", "Id")
|
|
require.True(t, ok)
|
|
|
|
err := sqlStore.GetMaster().SelectOne(&def, query)
|
|
require.NoError(t, err)
|
|
require.Nil(t, def)
|
|
})
|
|
|
|
t.Run("string default", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("Posts", "Id", model.NewString("'test'"), model.NewString("'test'"))
|
|
require.True(t, ok)
|
|
|
|
err := sqlStore.GetMaster().SelectOne(&def, query)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, def)
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
require.Equal(t, "test", *def)
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
require.Equal(t, "'test'::character varying", *def)
|
|
}
|
|
|
|
ok = sqlStore.RemoveDefaultIfColumnExists("Posts", "Id")
|
|
require.True(t, ok)
|
|
})
|
|
|
|
t.Run("int default", func(t *testing.T) {
|
|
ok := sqlStore.AlterDefaultIfColumnExists("Posts", "UpdateAt", model.NewString("0"), model.NewString("0"))
|
|
require.True(t, ok)
|
|
|
|
if sqlStore.DriverName() == model.DatabaseDriverMysql {
|
|
query = `SELECT column_default
|
|
FROM information_schema.columns
|
|
WHERE table_schema = DATABASE()
|
|
AND table_name = 'Posts'
|
|
AND column_name = 'UpdateAt'`
|
|
} else if sqlStore.DriverName() == model.DatabaseDriverPostgres {
|
|
query = `SELECT column_default
|
|
FROM information_schema.columns
|
|
WHERE table_name = 'posts'
|
|
AND column_name = 'updateat'`
|
|
}
|
|
|
|
err := sqlStore.GetMaster().SelectOne(&def, query)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, def)
|
|
require.Equal(t, "0", *def)
|
|
|
|
ok = sqlStore.RemoveDefaultIfColumnExists("Posts", "UpdateAt")
|
|
require.True(t, ok)
|
|
})
|
|
})
|
|
}
|