first commit
Some checks failed
Backend Tests / Static Checks (push) Has been cancelled
Backend Tests / Tests (other) (push) Has been cancelled
Backend Tests / Tests (plugin) (push) Has been cancelled
Backend Tests / Tests (server) (push) Has been cancelled
Backend Tests / Tests (store) (push) Has been cancelled
Build Canary Image / build-frontend (push) Has been cancelled
Build Canary Image / build-push (linux/amd64) (push) Has been cancelled
Build Canary Image / build-push (linux/arm64) (push) Has been cancelled
Build Canary Image / merge (push) Has been cancelled
Frontend Tests / Lint (push) Has been cancelled
Frontend Tests / Build (push) Has been cancelled
Proto Linter / Lint Protos (push) Has been cancelled
Some checks failed
Backend Tests / Static Checks (push) Has been cancelled
Backend Tests / Tests (other) (push) Has been cancelled
Backend Tests / Tests (plugin) (push) Has been cancelled
Backend Tests / Tests (server) (push) Has been cancelled
Backend Tests / Tests (store) (push) Has been cancelled
Build Canary Image / build-frontend (push) Has been cancelled
Build Canary Image / build-push (linux/amd64) (push) Has been cancelled
Build Canary Image / build-push (linux/arm64) (push) Has been cancelled
Build Canary Image / merge (push) Has been cancelled
Frontend Tests / Lint (push) Has been cancelled
Frontend Tests / Build (push) Has been cancelled
Proto Linter / Lint Protos (push) Has been cancelled
This commit is contained in:
200
store/test/migrator_test.go
Normal file
200
store/test/migrator_test.go
Normal file
@@ -0,0 +1,200 @@
|
||||
package test
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"os"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
|
||||
"github.com/usememos/memos/store"
|
||||
)
|
||||
|
||||
// TestFreshInstall verifies that LATEST.sql applies correctly on a fresh database.
|
||||
// This is essentially what NewTestingStore already does, but we make it explicit.
|
||||
func TestFreshInstall(t *testing.T) {
|
||||
t.Parallel()
|
||||
ctx := context.Background()
|
||||
|
||||
// NewTestingStore creates a fresh database and runs Migrate()
|
||||
// which applies LATEST.sql for uninitialized databases
|
||||
ts := NewTestingStore(ctx, t)
|
||||
|
||||
// Verify migration completed successfully
|
||||
currentSchemaVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
require.NotEmpty(t, currentSchemaVersion, "schema version should be set after fresh install")
|
||||
|
||||
// Verify we can read instance settings (basic sanity check)
|
||||
instanceSetting, err := ts.GetInstanceBasicSetting(ctx)
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, currentSchemaVersion, instanceSetting.SchemaVersion)
|
||||
}
|
||||
|
||||
// TestMigrationReRun verifies that re-running the migration on an already
|
||||
// migrated database does not fail or cause issues. This simulates a
|
||||
// scenario where the server is restarted.
|
||||
func TestMigrationReRun(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
ctx := context.Background()
|
||||
// Use the shared testing store which already runs migrations on init
|
||||
ts := NewTestingStore(ctx, t)
|
||||
|
||||
// Get current version
|
||||
initialVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
|
||||
// Manually trigger migration again
|
||||
err = ts.Migrate(ctx)
|
||||
require.NoError(t, err, "re-running migration should not fail")
|
||||
|
||||
// Verify version hasn't changed (or at least is valid)
|
||||
finalVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, initialVersion, finalVersion, "version should match after re-run")
|
||||
}
|
||||
|
||||
// TestMigrationWithData verifies that migration preserves data integrity.
|
||||
// Creates data, then re-runs migration and verifies data is still accessible.
|
||||
func TestMigrationWithData(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
ctx := context.Background()
|
||||
ts := NewTestingStore(ctx, t)
|
||||
|
||||
// Create a user and memo before re-running migration
|
||||
user, err := createTestingHostUser(ctx, ts)
|
||||
require.NoError(t, err, "should create user")
|
||||
|
||||
originalMemo, err := ts.CreateMemo(ctx, &store.Memo{
|
||||
UID: "migration-data-test",
|
||||
CreatorID: user.ID,
|
||||
Content: "Data before migration re-run",
|
||||
Visibility: store.Public,
|
||||
})
|
||||
require.NoError(t, err, "should create memo")
|
||||
|
||||
// Re-run migration
|
||||
err = ts.Migrate(ctx)
|
||||
require.NoError(t, err, "re-running migration should not fail")
|
||||
|
||||
// Verify data is still accessible
|
||||
memo, err := ts.GetMemo(ctx, &store.FindMemo{UID: &originalMemo.UID})
|
||||
require.NoError(t, err, "should retrieve memo after migration")
|
||||
require.Equal(t, "Data before migration re-run", memo.Content, "memo content should be preserved")
|
||||
}
|
||||
|
||||
// TestMigrationMultipleReRuns verifies that migration is idempotent
|
||||
// even when run multiple times in succession.
|
||||
func TestMigrationMultipleReRuns(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
ctx := context.Background()
|
||||
ts := NewTestingStore(ctx, t)
|
||||
|
||||
// Get initial version
|
||||
initialVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
|
||||
// Run migration multiple times
|
||||
for i := 0; i < 3; i++ {
|
||||
err = ts.Migrate(ctx)
|
||||
require.NoError(t, err, "migration run %d should not fail", i+1)
|
||||
}
|
||||
|
||||
// Verify version is still correct
|
||||
finalVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, initialVersion, finalVersion, "version should remain unchanged after multiple re-runs")
|
||||
}
|
||||
|
||||
// TestMigrationFromStableVersion verifies that upgrading from a stable Memos version
|
||||
// to the current version works correctly. This is the critical upgrade path test.
|
||||
//
|
||||
// Test flow:
|
||||
// 1. Start a stable Memos container to create a database with the old schema
|
||||
// 2. Stop the container and wait for cleanup
|
||||
// 3. Use the store directly to run migration with current code
|
||||
// 4. Verify the migration succeeded and data can be written
|
||||
//
|
||||
// Note: This test is skipped when running with -race flag because testcontainers
|
||||
// has known race conditions in its reaper code that are outside our control.
|
||||
func TestMigrationFromStableVersion(t *testing.T) {
|
||||
// Skip for non-SQLite drivers (simplifies the test)
|
||||
if getDriverFromEnv() != "sqlite" {
|
||||
t.Skip("skipping upgrade test for non-sqlite driver")
|
||||
}
|
||||
|
||||
// Skip if explicitly disabled (e.g., in environments without Docker)
|
||||
if os.Getenv("SKIP_CONTAINER_TESTS") == "1" {
|
||||
t.Skip("skipping container-based test (SKIP_CONTAINER_TESTS=1)")
|
||||
}
|
||||
|
||||
ctx := context.Background()
|
||||
dataDir := t.TempDir()
|
||||
|
||||
// 1. Start stable Memos container to create database with old schema
|
||||
cfg := MemosContainerConfig{
|
||||
Driver: "sqlite",
|
||||
DataDir: dataDir,
|
||||
Version: StableMemosVersion,
|
||||
}
|
||||
|
||||
t.Logf("Starting Memos %s container to create old-schema database...", cfg.Version)
|
||||
container, err := StartMemosContainer(ctx, cfg)
|
||||
require.NoError(t, err, "failed to start stable memos container")
|
||||
|
||||
// Wait for the container to fully initialize the database
|
||||
time.Sleep(10 * time.Second)
|
||||
|
||||
// Stop the container gracefully
|
||||
t.Log("Stopping stable Memos container...")
|
||||
err = container.Terminate(ctx)
|
||||
require.NoError(t, err, "failed to stop memos container")
|
||||
|
||||
// Wait for file handles to be released
|
||||
time.Sleep(2 * time.Second)
|
||||
|
||||
// 2. Connect to the database directly and run migration with current code
|
||||
dsn := fmt.Sprintf("%s/memos_prod.db", dataDir)
|
||||
t.Logf("Connecting to database at %s...", dsn)
|
||||
|
||||
ts := NewTestingStoreWithDSN(ctx, t, "sqlite", dsn)
|
||||
|
||||
// Get the schema version before migration
|
||||
oldSetting, err := ts.GetInstanceBasicSetting(ctx)
|
||||
require.NoError(t, err)
|
||||
t.Logf("Old schema version: %s", oldSetting.SchemaVersion)
|
||||
|
||||
// 3. Run migration with current code
|
||||
t.Log("Running migration with current code...")
|
||||
err = ts.Migrate(ctx)
|
||||
require.NoError(t, err, "migration from stable version should succeed")
|
||||
|
||||
// 4. Verify migration succeeded
|
||||
newVersion, err := ts.GetCurrentSchemaVersion()
|
||||
require.NoError(t, err)
|
||||
t.Logf("New schema version: %s", newVersion)
|
||||
|
||||
newSetting, err := ts.GetInstanceBasicSetting(ctx)
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, newVersion, newSetting.SchemaVersion, "schema version should be updated")
|
||||
|
||||
// Verify we can write data to the migrated database
|
||||
user, err := createTestingHostUser(ctx, ts)
|
||||
require.NoError(t, err, "should create user after migration")
|
||||
|
||||
memo, err := ts.CreateMemo(ctx, &store.Memo{
|
||||
UID: "post-upgrade-memo",
|
||||
CreatorID: user.ID,
|
||||
Content: "Content after upgrade from stable",
|
||||
Visibility: store.Public,
|
||||
})
|
||||
require.NoError(t, err, "should create memo after migration")
|
||||
require.Equal(t, "Content after upgrade from stable", memo.Content)
|
||||
|
||||
t.Logf("Migration successful: %s -> %s", oldSetting.SchemaVersion, newVersion)
|
||||
}
|
||||
Reference in New Issue
Block a user