2021-11-11 17:01:43 +01:00
|
|
|
package sqlstore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/mattermost/focalboard/server/model"
|
|
|
|
|
2022-10-15 00:39:25 +02:00
|
|
|
"github.com/stretchr/testify/assert"
|
2021-11-11 17:01:43 +01:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestGetBlocksWithSameID(t *testing.T) {
|
2022-03-22 15:24:34 +01:00
|
|
|
t.Skip("we need to setup a test with the database migrated up to version 14 and then run these tests")
|
|
|
|
|
2021-11-11 17:01:43 +01:00
|
|
|
store, tearDown := SetupTests(t)
|
|
|
|
sqlStore := store.(*SQLStore)
|
|
|
|
defer tearDown()
|
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
container1 := "1"
|
|
|
|
container2 := "2"
|
|
|
|
container3 := "3"
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
block1 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block2 := &model.Block{ID: "block-id-2", BoardID: "board-id-2"}
|
|
|
|
block3 := &model.Block{ID: "block-id-3", BoardID: "board-id-3"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
block4 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block5 := &model.Block{ID: "block-id-2", BoardID: "board-id-2"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
block6 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block7 := &model.Block{ID: "block-id-7", BoardID: "board-id-7"}
|
|
|
|
block8 := &model.Block{ID: "block-id-8", BoardID: "board-id-8"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block1, block2, block3} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container1, block, "user-id")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block4, block5} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container2, block, "user-id")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block6, block7, block8} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container3, block, "user-id")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
blocksWithDuplicatedID := []*model.Block{block1, block2, block4, block5, block6}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
blocks, err := sqlStore.getBlocksWithSameID(sqlStore.db)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// we process the found blocks to remove extra information and be
|
|
|
|
// able to compare both expected and found sets
|
2022-10-25 22:46:43 +02:00
|
|
|
foundBlocks := []*model.Block{}
|
2021-11-11 17:01:43 +01:00
|
|
|
for _, foundBlock := range blocks {
|
2022-10-25 22:46:43 +02:00
|
|
|
foundBlocks = append(foundBlocks, &model.Block{ID: foundBlock.ID, BoardID: foundBlock.BoardID})
|
2021-11-11 17:01:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
require.ElementsMatch(t, blocksWithDuplicatedID, foundBlocks)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestReplaceBlockID(t *testing.T) {
|
2022-03-22 15:24:34 +01:00
|
|
|
t.Skip("we need to setup a test with the database migrated up to version 14 and then run these tests")
|
|
|
|
|
2021-11-11 17:01:43 +01:00
|
|
|
store, tearDown := SetupTests(t)
|
|
|
|
sqlStore := store.(*SQLStore)
|
|
|
|
defer tearDown()
|
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
container1 := "1"
|
|
|
|
container2 := "2"
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
// blocks from team1
|
2022-10-25 22:46:43 +02:00
|
|
|
block1 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block2 := &model.Block{ID: "block-id-2", BoardID: "board-id-2", ParentID: "block-id-1"}
|
|
|
|
block3 := &model.Block{ID: "block-id-3", BoardID: "block-id-1"}
|
|
|
|
block4 := &model.Block{ID: "block-id-4", BoardID: "block-id-2"}
|
|
|
|
block5 := &model.Block{ID: "block-id-5", BoardID: "block-id-1", ParentID: "block-id-1"}
|
|
|
|
block8 := &model.Block{
|
2022-03-22 15:24:34 +01:00
|
|
|
ID: "block-id-8", BoardID: "board-id-2", Type: model.TypeCard,
|
2021-12-10 15:33:41 +01:00
|
|
|
Fields: map[string]interface{}{"contentOrder": []string{"block-id-1", "block-id-2"}},
|
|
|
|
}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
// blocks from team2. They're identical to blocks 1 and 2,
|
2021-11-11 17:01:43 +01:00
|
|
|
// but they shouldn't change
|
2022-10-25 22:46:43 +02:00
|
|
|
block6 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block7 := &model.Block{ID: "block-id-2", BoardID: "board-id-2", ParentID: "block-id-1"}
|
|
|
|
block9 := &model.Block{
|
2022-03-22 15:24:34 +01:00
|
|
|
ID: "block-id-8", BoardID: "board-id-2", Type: model.TypeCard,
|
2021-12-10 15:33:41 +01:00
|
|
|
Fields: map[string]interface{}{"contentOrder": []string{"block-id-1", "block-id-2"}},
|
|
|
|
}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block1, block2, block3, block4, block5, block8} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container1, block, "user-id")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block6, block7, block9} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container2, block, "user-id")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
|
|
|
currentID := "block-id-1"
|
|
|
|
newID := "new-id-1"
|
|
|
|
err := sqlStore.replaceBlockID(sqlStore.db, currentID, newID, "1")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock1, err := sqlStore.getLegacyBlock(sqlStore.db, container1, newID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock2, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block2.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock3, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block3.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock5, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block5.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock6, err := sqlStore.getLegacyBlock(sqlStore.db, container2, block6.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock7, err := sqlStore.getLegacyBlock(sqlStore.db, container2, block7.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock8, err := sqlStore.GetBlock(block8.ID)
|
2021-12-10 15:33:41 +01:00
|
|
|
require.NoError(t, err)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock9, err := sqlStore.GetBlock(block9.ID)
|
2021-12-10 15:33:41 +01:00
|
|
|
require.NoError(t, err)
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
require.Equal(t, newID, newBlock1.ID)
|
|
|
|
require.Equal(t, newID, newBlock2.ParentID)
|
2022-03-22 15:24:34 +01:00
|
|
|
require.Equal(t, newID, newBlock3.BoardID)
|
|
|
|
require.Equal(t, newID, newBlock5.BoardID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.Equal(t, newID, newBlock5.ParentID)
|
2021-12-10 15:33:41 +01:00
|
|
|
require.Equal(t, newBlock8.Fields["contentOrder"].([]interface{})[0], newID)
|
|
|
|
require.Equal(t, newBlock8.Fields["contentOrder"].([]interface{})[1], "block-id-2")
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
require.Equal(t, currentID, newBlock6.ID)
|
|
|
|
require.Equal(t, currentID, newBlock7.ParentID)
|
2021-12-10 15:33:41 +01:00
|
|
|
require.Equal(t, newBlock9.Fields["contentOrder"].([]interface{})[0], "block-id-1")
|
|
|
|
require.Equal(t, newBlock9.Fields["contentOrder"].([]interface{})[1], "block-id-2")
|
2021-11-11 17:01:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRunUniqueIDsMigration(t *testing.T) {
|
2022-03-22 15:24:34 +01:00
|
|
|
t.Skip("we need to setup a test with the database migrated up to version 14 and then run these tests")
|
|
|
|
|
2021-11-11 17:01:43 +01:00
|
|
|
store, tearDown := SetupTests(t)
|
|
|
|
sqlStore := store.(*SQLStore)
|
|
|
|
defer tearDown()
|
|
|
|
|
|
|
|
// we need to mark the migration as not done so we can run it
|
|
|
|
// again with the test data
|
|
|
|
keyErr := sqlStore.SetSystemSetting(UniqueIDsMigrationKey, "false")
|
|
|
|
require.NoError(t, keyErr)
|
|
|
|
|
2022-03-22 15:24:34 +01:00
|
|
|
container1 := "1"
|
|
|
|
container2 := "2"
|
|
|
|
container3 := "3"
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
// blocks from workspace1. They shouldn't change, as the first
|
|
|
|
// duplicated ID is preserved
|
2022-10-25 22:46:43 +02:00
|
|
|
block1 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block2 := &model.Block{ID: "block-id-2", BoardID: "board-id-2", ParentID: "block-id-1"}
|
|
|
|
block3 := &model.Block{ID: "block-id-3", BoardID: "block-id-1"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
// blocks from workspace2. They're identical to blocks 1, 2 and 3,
|
|
|
|
// and they should change
|
2022-10-25 22:46:43 +02:00
|
|
|
block4 := &model.Block{ID: "block-id-1", BoardID: "board-id-1"}
|
|
|
|
block5 := &model.Block{ID: "block-id-2", BoardID: "board-id-2", ParentID: "block-id-1"}
|
|
|
|
block6 := &model.Block{ID: "block-id-6", BoardID: "block-id-1", ParentID: "block-id-2"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
// block from workspace3. It should change as well
|
2022-10-25 22:46:43 +02:00
|
|
|
block7 := &model.Block{ID: "block-id-2", BoardID: "board-id-2"}
|
2021-11-11 17:01:43 +01:00
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block1, block2, block3} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container1, block, "user-id-2")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block4, block5, block6} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container2, block, "user-id-2")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-10-25 22:46:43 +02:00
|
|
|
for _, block := range []*model.Block{block7} {
|
|
|
|
err := sqlStore.insertLegacyBlock(sqlStore.db, container3, block, "user-id-2")
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2022-08-18 17:03:33 +02:00
|
|
|
err := sqlStore.RunUniqueIDsMigration()
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// blocks from workspace 1 haven't changed, so we can simply fetch them
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock1, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block1.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock1)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock2, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block2.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock2)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock3, err := sqlStore.getLegacyBlock(sqlStore.db, container1, block3.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock3)
|
|
|
|
|
|
|
|
// first two blocks from workspace 2 have changed, so we fetch
|
|
|
|
// them through the third one, which points to the new IDs
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock6, err := sqlStore.getLegacyBlock(sqlStore.db, container2, block6.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock6)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock4, err := sqlStore.getLegacyBlock(sqlStore.db, container2, newBlock6.BoardID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock4)
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock5, err := sqlStore.getLegacyBlock(sqlStore.db, container2, newBlock6.ParentID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.NotNil(t, newBlock5)
|
|
|
|
|
|
|
|
// block from workspace 3 changed as well, so we shouldn't be able
|
|
|
|
// to fetch it
|
2022-03-22 15:24:34 +01:00
|
|
|
newBlock7, err := sqlStore.getLegacyBlock(sqlStore.db, container3, block7.ID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Nil(t, newBlock7)
|
|
|
|
|
|
|
|
// workspace 1 block links are maintained
|
|
|
|
require.Equal(t, newBlock1.ID, newBlock2.ParentID)
|
2022-03-22 15:24:34 +01:00
|
|
|
require.Equal(t, newBlock1.ID, newBlock3.BoardID)
|
2021-11-11 17:01:43 +01:00
|
|
|
|
|
|
|
// workspace 2 first two block IDs have changed
|
2022-03-22 15:24:34 +01:00
|
|
|
require.NotEqual(t, block4.ID, newBlock4.BoardID)
|
2021-11-11 17:01:43 +01:00
|
|
|
require.NotEqual(t, block5.ID, newBlock5.ParentID)
|
|
|
|
}
|
2022-10-15 00:39:25 +02:00
|
|
|
|
|
|
|
func TestCheckForMismatchedCollation(t *testing.T) {
|
|
|
|
store, tearDown := SetupTests(t)
|
|
|
|
sqlStore := store.(*SQLStore)
|
|
|
|
defer tearDown()
|
|
|
|
|
|
|
|
if sqlStore.dbType != model.MysqlDBType {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure all collations are consistent.
|
|
|
|
tableNames, err := sqlStore.getFocalBoardTableNames()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
sqlCollation := "SELECT table_collation FROM information_schema.tables WHERE table_name=? and table_schema=(SELECT DATABASE())"
|
|
|
|
stmtCollation, err := sqlStore.db.Prepare(sqlCollation)
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer stmtCollation.Close()
|
|
|
|
|
|
|
|
var collation string
|
|
|
|
|
|
|
|
// make sure the correct charset is applied to each table.
|
|
|
|
for i, name := range tableNames {
|
|
|
|
row := stmtCollation.QueryRow(name)
|
|
|
|
|
|
|
|
var actualCollation string
|
|
|
|
err = row.Scan(&actualCollation)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
if collation == "" {
|
|
|
|
collation = actualCollation
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.Equalf(t, collation, actualCollation, "for table_name='%s', index=%d", name, i)
|
|
|
|
}
|
|
|
|
}
|