From 457aa4c9d399de0586af1a7ab4b7afff375b72a9 Mon Sep 17 00:00:00 2001 From: Dan Hansen Date: Thu, 11 Apr 2024 10:18:16 -0700 Subject: [PATCH] Use `WITHOUT ROWID` table clustering for primary keys (#44) * Use `WITHOUT ROWID` table clustering for tables * lint --- driver_test.go | 31 +++++++++++++++++++++++++++++++ internal/spec.go | 16 ++++++++++++++-- 2 files changed, 45 insertions(+), 2 deletions(-) diff --git a/driver_test.go b/driver_test.go index 2f56048..f346a04 100644 --- a/driver_test.go +++ b/driver_test.go @@ -3,6 +3,7 @@ package zetasqlite_test import ( "context" "database/sql" + "strings" "testing" zetasqlite "github.com/goccy/go-zetasqlite" @@ -179,6 +180,36 @@ CREATE TABLE IF NOT EXISTS Singers ( }) } +func TestCreateTable(t *testing.T) { + t.Run("primary keys", func(t *testing.T) { + db, err := sql.Open("zetasqlite", ":memory:") + if err != nil { + t.Fatal(err) + } + if _, err := db.Exec(` +CREATE TABLE IF NOT EXISTS Singers ( + SingerId INT64 NOT NULL PRIMARY KEY, + FirstName STRING(1024), + LastName STRING(1024) +)`); err != nil { + t.Fatal(err) + } + stmt, err := db.Prepare("INSERT Singers (SingerId, FirstName, LastName) VALUES (@SingerID, @FirstName, @LastName)") + if err != nil { + t.Fatal(err) + } + _, err = stmt.Exec(int64(1), "Kylie", "Minogue") + if err != nil { + t.Fatal(err) + } + + _, err = stmt.Exec(int64(1), "Miss", "Kitten") + if !strings.HasSuffix(err.Error(), "UNIQUE constraint failed: Singers.SingerId") { + t.Fatalf("expected failed unique constraint err, got: %s", err) + } + }) +} + func TestPreparedStatements(t *testing.T) { t.Run("prepared select", func(t *testing.T) { db, err := sql.Open("zetasqlite", ":memory:") diff --git a/internal/spec.go b/internal/spec.go index c343f9e..41fd868 100644 --- a/internal/spec.go +++ b/internal/spec.go @@ -139,11 +139,23 @@ func (s *TableSpec) SQLiteSchema() string { columns = append(columns, c.SQLiteSchema()) } if len(s.PrimaryKey) != 0 { + primaryKeys := make([]string, len(s.PrimaryKey)) + + for i, key := range s.PrimaryKey { + primaryKeys[i] = fmt.Sprintf("%s COLLATE zetasqlite_collate", key) + } + columns = append( columns, - fmt.Sprintf("PRIMARY KEY (%s)", strings.Join(s.PrimaryKey, ",")), + fmt.Sprintf("PRIMARY KEY (%s)", strings.Join(primaryKeys, ",")), ) } + var clustering string + if len(s.PrimaryKey) > 0 { + clustering = "WITHOUT ROWID" + } else { + clustering = "" + } var stmt string switch s.CreateMode { case ast.CreateDefaultMode: @@ -153,7 +165,7 @@ func (s *TableSpec) SQLiteSchema() string { case ast.CreateIfNotExistsMode: stmt = "CREATE TABLE IF NOT EXISTS" } - return fmt.Sprintf("%s `%s` (%s)", stmt, s.TableName(), strings.Join(columns, ",")) + return fmt.Sprintf("%s `%s` (%s) %s", stmt, s.TableName(), strings.Join(columns, ","), clustering) } func viewSQLiteSchema(s *TableSpec) string {