package db import ( "database/sql" "fmt" "os" "path" "testing" "time" ) func Test_IsValidSQLiteOnDisk(t *testing.T) { path := mustTempFile() defer os.Remove(path) dsn := fmt.Sprintf("file:%s", path) db, err := sql.Open("sqlite3", dsn) if err != nil { t.Fatalf("failed to create SQLite database: %s", err.Error()) } _, err = db.Exec("CREATE TABLE foo (name TEXT)") if err != nil { t.Fatalf("failed to create table: %s", err.Error()) } if err := db.Close(); err != nil { t.Fatalf("failed to close database: %s", err.Error()) } if !IsValidSQLiteFile(path) { t.Fatalf("good SQLite file marked as invalid") } data, err := os.ReadFile(path) if err != nil { t.Fatalf("failed to read SQLite file: %s", err.Error()) } if !IsValidSQLiteData(data) { t.Fatalf("good SQLite data marked as invalid") } } func Test_IsWALModeEnablednDiskDELETE(t *testing.T) { path := mustTempFile() defer os.Remove(path) dsn := fmt.Sprintf("file:%s", path) db, err := sql.Open("sqlite3", dsn) if err != nil { t.Fatalf("failed to create SQLite database: %s", err.Error()) } _, err = db.Exec("CREATE TABLE foo (name TEXT)") if err != nil { t.Fatalf("failed to create table: %s", err.Error()) } if err := db.Close(); err != nil { t.Fatalf("failed to close database: %s", err.Error()) } if IsWALModeEnabledSQLiteFile(path) { t.Fatalf("non WAL file marked as WAL") } data, err := os.ReadFile(path) if err != nil { t.Fatalf("failed to read SQLite file: %s", err.Error()) } if IsWALModeEnabled(data) { t.Fatalf("non WAL data marked as WAL") } } func Test_IsWALModeEnablednDiskWAL(t *testing.T) { path := mustTempFile() defer os.Remove(path) dsn := fmt.Sprintf("file:%s", path) db, err := sql.Open("sqlite3", dsn) if err != nil { t.Fatalf("failed to create SQLite database: %s", err.Error()) } _, err = db.Exec("CREATE TABLE foo (name TEXT)") if err != nil { t.Fatalf("failed to create table: %s", err.Error()) } _, err = db.Exec("PRAGMA journal_mode=WAL") if err != nil { t.Fatalf("failed to enable WAL mode: %s", err.Error()) } if err := db.Close(); err != nil { t.Fatalf("failed to close database: %s", err.Error()) } if !IsWALModeEnabledSQLiteFile(path) { t.Fatalf("WAL file marked as non-WAL") } data, err := os.ReadFile(path) if err != nil { t.Fatalf("failed to read SQLite file: %s", err.Error()) } if !IsWALModeEnabled(data) { t.Fatalf("WAL data marked as non-WAL") } } func Test_FileCreationOnDisk(t *testing.T) { dir := t.TempDir() dbPath := path.Join(dir, "test_db") db, err := Open(dbPath, false) if err != nil { t.Fatalf("failed to open new database: %s", err.Error()) } if db == nil { t.Fatal("database is nil") } defer db.Close() if db.InMemory() { t.Fatal("on-disk database marked as in-memory") } if db.FKEnabled() { t.Fatal("FK constraints marked as enabled") } if db.Path() != dbPath { t.Fatal("database path is incorrect") } // Confirm checkpoint works without error on a database in DELETE Mode. // It's just ignored. if err := db.Checkpoint(5 * time.Second); err != nil { t.Fatalf("failed to checkpoint database in DELETE mode: %s", err.Error()) } if _, err := os.Stat(dbPath); os.IsNotExist(err) { t.Fatalf("%s does not exist after open", dbPath) } err = db.Close() if err != nil { t.Fatalf("failed to close database: %s", err.Error()) } } // Test_ConnectionIsolationOnDisk test that ISOLATION behavior of on-disk databases doesn't // change unexpectedly. func Test_ConnectionIsolationOnDisk(t *testing.T) { dir := t.TempDir() dbPath := path.Join(dir, "test_db") db, err := Open(dbPath, false) if err != nil { t.Fatalf("failed to open new database: %s", err.Error()) } defer db.Close() r, err := db.ExecuteStringStmt("CREATE TABLE foo (id INTEGER NOT NULL PRIMARY KEY, name TEXT)") if err != nil { t.Fatalf("failed to create table: %s", err.Error()) } if exp, got := `[{}]`, asJSON(r); exp != got { t.Fatalf("unexpected results for query, expected %s, got %s", exp, got) } r, err = db.ExecuteStringStmt("BEGIN") if err != nil { t.Fatalf("failed to create table: %s", err.Error()) } if exp, got := `[{}]`, asJSON(r); exp != got { t.Fatalf("unexpected results for query, expected %s, got %s", exp, got) } r, err = db.ExecuteStringStmt(`INSERT INTO foo(name) VALUES("fiona")`) if err != nil { t.Fatalf("error executing insertion into table: %s", err.Error()) } if exp, got := `[{"last_insert_id":1,"rows_affected":1}]`, asJSON(r); exp != got { t.Fatalf("unexpected results for execute, expected %s, got %s", exp, got) } q, err := db.QueryStringStmt("SELECT * FROM foo") if err != nil { t.Fatalf("failed to query empty table: %s", err.Error()) } if exp, got := `[{"columns":["id","name"],"types":["integer","text"]}]`, asJSON(q); exp != got { t.Fatalf("unexpected results for query, expected %s, got %s", exp, got) } _, err = db.ExecuteStringStmt("COMMIT") if err != nil { t.Fatalf("error executing insertion into table: %s", err.Error()) } q, err = db.QueryStringStmt("SELECT * FROM foo") if err != nil { t.Fatalf("failed to query empty table: %s", err.Error()) } if exp, got := `[{"columns":["id","name"],"types":["integer","text"],"values":[[1,"fiona"]]}]`, asJSON(q); exp != got { t.Fatalf("unexpected results for query, expected %s, got %s", exp, got) } }