mirror of
https://github.com/sipeed/picoclaw.git
synced 2026-05-25 16:00:35 +00:00
b7db059544
* feat(chat,seahorse): persist and display model_name across history * test(seahorse): fix lint regressions in repair coverage * fix(pico): preserve model_name in live updates * fix(pico): preserve model_name through live stream wrappers
380 lines
10 KiB
Go
380 lines
10 KiB
Go
package seahorse
|
|
|
|
import (
|
|
"database/sql"
|
|
"fmt"
|
|
"strings"
|
|
"sync/atomic"
|
|
"testing"
|
|
|
|
_ "modernc.org/sqlite"
|
|
)
|
|
|
|
var testDBCounter uint64
|
|
|
|
func openTestDB(t *testing.T) *sql.DB {
|
|
t.Helper()
|
|
|
|
n := atomic.AddUint64(&testDBCounter, 1)
|
|
testName := strings.NewReplacer("/", "_", " ", "_").Replace(t.Name())
|
|
// Use a shared in-memory database so concurrent goroutines/connections in tests
|
|
// observe the same schema/data.
|
|
dsn := fmt.Sprintf("file:seahorse_test_%s_%d?mode=memory&cache=shared", testName, n)
|
|
|
|
db, err := sql.Open("sqlite", dsn)
|
|
if err != nil {
|
|
t.Fatalf("open test db: %v", err)
|
|
}
|
|
t.Cleanup(func() { db.Close() })
|
|
return db
|
|
}
|
|
|
|
func TestRunMigrations(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("runSchema: %v", err)
|
|
}
|
|
|
|
// Verify all tables exist
|
|
tables := []string{
|
|
"conversations",
|
|
"messages",
|
|
"message_parts",
|
|
"summaries",
|
|
"summary_parents",
|
|
"summary_messages",
|
|
"context_items",
|
|
}
|
|
for _, tbl := range tables {
|
|
var name string
|
|
err := db.QueryRow(
|
|
"SELECT name FROM sqlite_master WHERE type='table' AND name=?", tbl,
|
|
).Scan(&name)
|
|
if err != nil {
|
|
t.Errorf("table %q not found: %v", tbl, err)
|
|
}
|
|
}
|
|
|
|
// Verify FTS5 virtual table exists
|
|
var ftsName string
|
|
err := db.QueryRow(
|
|
"SELECT name FROM sqlite_master WHERE type='table' AND name='summaries_fts'",
|
|
).Scan(&ftsName)
|
|
if err != nil {
|
|
t.Errorf("FTS5 table summaries_fts not found: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestRunMigrationsIdempotent(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
// Run migrations twice — should succeed both times
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("first migration: %v", err)
|
|
}
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("second migration (idempotent): %v", err)
|
|
}
|
|
|
|
// Verify we can still insert data after double migration
|
|
res, err := db.Exec(
|
|
"INSERT INTO conversations (session_key, created_at, updated_at) VALUES (?, datetime('now'), datetime('now'))",
|
|
"test-session",
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("insert after double migration: %v", err)
|
|
}
|
|
id, _ := res.LastInsertId()
|
|
if id == 0 {
|
|
t.Error("expected non-zero conversation id")
|
|
}
|
|
}
|
|
|
|
func TestRunSchemaAddsMessagesReasoningContentColumn(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
_, err := db.Exec(`CREATE TABLE messages (
|
|
message_id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
conversation_id INTEGER NOT NULL,
|
|
role TEXT NOT NULL,
|
|
content TEXT NOT NULL DEFAULT '',
|
|
token_count INTEGER NOT NULL DEFAULT 0,
|
|
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
)`)
|
|
if err != nil {
|
|
t.Fatalf("create legacy messages table: %v", err)
|
|
}
|
|
|
|
err = runSchema(db)
|
|
if err != nil {
|
|
t.Fatalf("runSchema: %v", err)
|
|
}
|
|
|
|
var count int
|
|
err = db.QueryRow(`SELECT count(*) FROM pragma_table_info('messages') WHERE name = 'reasoning_content'`).
|
|
Scan(&count)
|
|
if err != nil {
|
|
t.Fatalf("query pragma_table_info: %v", err)
|
|
}
|
|
if count != 1 {
|
|
t.Fatalf("reasoning_content column count = %d, want 1", count)
|
|
}
|
|
|
|
_, err = db.Exec(
|
|
`INSERT INTO conversations (session_key, created_at, updated_at) VALUES (?, datetime('now'), datetime('now'))`,
|
|
"reasoning-column-test",
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("insert conversation: %v", err)
|
|
}
|
|
|
|
_, err = db.Exec(
|
|
`INSERT INTO messages (conversation_id, role, content, reasoning_content, token_count)
|
|
VALUES (1, 'assistant', 'answer', 'thinking', 1)`,
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("insert message with reasoning_content: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestRunSchemaAddsMessagesModelNameColumn(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
_, err := db.Exec(`CREATE TABLE messages (
|
|
message_id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
conversation_id INTEGER NOT NULL,
|
|
role TEXT NOT NULL,
|
|
content TEXT NOT NULL DEFAULT '',
|
|
reasoning_content TEXT NOT NULL DEFAULT '',
|
|
token_count INTEGER NOT NULL DEFAULT 0,
|
|
created_at TEXT NOT NULL DEFAULT (datetime('now'))
|
|
)`)
|
|
if err != nil {
|
|
t.Fatalf("create legacy messages table: %v", err)
|
|
}
|
|
|
|
err = runSchema(db)
|
|
if err != nil {
|
|
t.Fatalf("runSchema: %v", err)
|
|
}
|
|
|
|
var count int
|
|
err = db.QueryRow(`SELECT count(*) FROM pragma_table_info('messages') WHERE name = 'model_name'`).Scan(&count)
|
|
if err != nil {
|
|
t.Fatalf("query pragma_table_info: %v", err)
|
|
}
|
|
if count != 1 {
|
|
t.Fatalf("model_name column count = %d, want 1", count)
|
|
}
|
|
}
|
|
|
|
func TestMigrationConversationUnique(t *testing.T) {
|
|
db := openTestDB(t)
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("migration: %v", err)
|
|
}
|
|
|
|
// Insert first
|
|
_, err := db.Exec(
|
|
"INSERT INTO conversations (session_key, created_at, updated_at) VALUES (?, datetime('now'), datetime('now'))",
|
|
"unique-key",
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("first insert: %v", err)
|
|
}
|
|
|
|
// Duplicate should fail
|
|
_, err = db.Exec(
|
|
"INSERT INTO conversations (session_key, created_at, updated_at) VALUES (?, datetime('now'), datetime('now'))",
|
|
"unique-key",
|
|
)
|
|
if err == nil {
|
|
t.Error("expected unique constraint violation for duplicate session_key")
|
|
}
|
|
}
|
|
|
|
func TestMigrationSummaryFTSInsert(t *testing.T) {
|
|
db := openTestDB(t)
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("migration: %v", err)
|
|
}
|
|
|
|
// Insert a conversation first
|
|
_, err := db.Exec(
|
|
"INSERT INTO conversations (session_key, created_at, updated_at) VALUES (?, datetime('now'), datetime('now'))",
|
|
"fts-test",
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("insert conversation: %v", err)
|
|
}
|
|
|
|
// Insert a summary
|
|
_, err = db.Exec(
|
|
`INSERT INTO summaries (summary_id, conversation_id, kind, depth, content, token_count, created_at)
|
|
VALUES ('sum_test1', 1, 'leaf', 0, '你好世界 hello world', 10, datetime('now'))`)
|
|
if err != nil {
|
|
t.Fatalf("insert summary: %v", err)
|
|
}
|
|
|
|
// FTS should find it — trigram tokenizer requires >= 3 chars
|
|
rows, err := db.Query(
|
|
"SELECT summary_id FROM summaries_fts WHERE summaries_fts MATCH ?",
|
|
"你好世",
|
|
)
|
|
if err != nil {
|
|
t.Fatalf("FTS query: %v", err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
var found string
|
|
if rows.Next() {
|
|
if err := rows.Scan(&found); err != nil {
|
|
t.Fatalf("scan: %v", err)
|
|
}
|
|
}
|
|
if err := rows.Err(); err != nil {
|
|
t.Fatalf("rows.Err: %v", err)
|
|
}
|
|
if found != "sum_test1" {
|
|
t.Errorf("FTS: expected 'sum_test1', got %q", found)
|
|
}
|
|
}
|
|
|
|
func TestMigrationSummaryParentsPK(t *testing.T) {
|
|
db := openTestDB(t)
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("migration: %v", err)
|
|
}
|
|
|
|
// Insert two summaries
|
|
for _, id := range []string{"sum_a", "sum_b"} {
|
|
_, err := db.Exec(
|
|
`INSERT INTO summaries (summary_id, conversation_id, kind, depth, content, token_count, created_at)
|
|
VALUES (?, 1, 'leaf', 0, 'content', 5, datetime('now'))`, id)
|
|
if err != nil {
|
|
t.Fatalf("insert summary %s: %v", id, err)
|
|
}
|
|
}
|
|
|
|
// Link child to parent
|
|
_, err := db.Exec(
|
|
"INSERT INTO summary_parents (summary_id, parent_summary_id) VALUES ('sum_a', 'sum_b')")
|
|
if err != nil {
|
|
t.Fatalf("link: %v", err)
|
|
}
|
|
|
|
// Duplicate link should fail (composite PK)
|
|
_, err = db.Exec(
|
|
"INSERT INTO summary_parents (summary_id, parent_summary_id) VALUES ('sum_a', 'sum_b')")
|
|
if err == nil {
|
|
t.Error("expected unique constraint violation for duplicate summary_parents link")
|
|
}
|
|
}
|
|
|
|
func TestTriggerMigration(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
// Run schema once to create tables and (correct) triggers
|
|
if err := runSchema(db); err != nil {
|
|
t.Fatalf("runSchema: %v", err)
|
|
}
|
|
|
|
// Drop correct triggers and recreate them with the old buggy body.
|
|
// The old trigger used INSERT INTO fts VALUES('delete', ...) which is wrong
|
|
// for non-external-content FTS5 tables.
|
|
oldSummariesDelete := `CREATE TRIGGER summaries_ad AFTER DELETE ON summaries BEGIN
|
|
INSERT INTO summaries_fts (summaries_fts, summary_id, content) VALUES('delete', old.summary_id, old.content);
|
|
END`
|
|
oldMessagesDelete := `CREATE TRIGGER messages_ad AFTER DELETE ON messages BEGIN
|
|
INSERT INTO messages_fts (messages_fts, message_id, content) VALUES('delete', old.message_id, old.content);
|
|
END`
|
|
|
|
for _, sql := range []string{
|
|
`DROP TRIGGER IF EXISTS summaries_ad`,
|
|
`DROP TRIGGER IF EXISTS messages_ad`,
|
|
oldSummariesDelete,
|
|
oldMessagesDelete,
|
|
} {
|
|
if _, err := db.Exec(sql); err != nil {
|
|
t.Fatalf("setup old trigger: %v", err)
|
|
}
|
|
}
|
|
|
|
// Insert a conversation and summary so we have something to delete
|
|
_, err := db.Exec(`INSERT INTO conversations (session_key) VALUES ('old-db-test')`)
|
|
if err != nil {
|
|
t.Fatalf("insert conversation: %v", err)
|
|
}
|
|
_, err = db.Exec(`INSERT INTO summaries (summary_id, conversation_id, kind, depth, content, token_count)
|
|
VALUES ('old-sum', 1, 'leaf', 0, 'old content', 5)`)
|
|
if err != nil {
|
|
t.Fatalf("insert summary: %v", err)
|
|
}
|
|
|
|
// The old trigger body is wrong for normal FTS5 — DELETE should fail.
|
|
_, err = db.Exec(`DELETE FROM summaries WHERE summary_id = 'old-sum'`)
|
|
if err == nil {
|
|
t.Error("expected error from old buggy trigger, but DELETE succeeded")
|
|
} else {
|
|
t.Logf("old trigger correctly causes error: %v", err)
|
|
}
|
|
|
|
// Now runSchema again — this drops and recreates the triggers with correct bodies.
|
|
err = runSchema(db)
|
|
if err != nil {
|
|
t.Fatalf("runSchema migration: %v", err)
|
|
}
|
|
|
|
// Insert again so we have data to delete
|
|
_, err = db.Exec(`INSERT INTO summaries (summary_id, conversation_id, kind, depth, content, token_count)
|
|
VALUES ('migrated-sum', 1, 'leaf', 0, 'new content', 5)`)
|
|
if err != nil {
|
|
t.Fatalf("insert after migration: %v", err)
|
|
}
|
|
|
|
// DELETE should now work with the corrected trigger body.
|
|
_, err = db.Exec(`DELETE FROM summaries WHERE summary_id = 'migrated-sum'`)
|
|
if err != nil {
|
|
t.Fatalf("DELETE after migration failed (trigger not corrected): %v", err)
|
|
}
|
|
|
|
// Verify the summary is gone
|
|
var count int
|
|
err = db.QueryRow(`SELECT count(*) FROM summaries WHERE summary_id = 'migrated-sum'`).Scan(&count)
|
|
if err != nil {
|
|
t.Fatalf("query after delete: %v", err)
|
|
}
|
|
if count != 0 {
|
|
t.Errorf("summary should be gone after DELETE, got count=%d", count)
|
|
}
|
|
}
|
|
|
|
func TestFTS5SQLConstants(t *testing.T) {
|
|
db := openTestDB(t)
|
|
|
|
// Verify FTS5 check SQL executes without error
|
|
_, err := db.Exec(sqlCheckFTS5Available)
|
|
if err != nil {
|
|
t.Errorf("sqlCheckFTS5Available failed: %v", err)
|
|
}
|
|
|
|
// Verify trigram check SQL executes without error
|
|
_, err = db.Exec(sqlCheckTrigramAvailable)
|
|
if err != nil {
|
|
t.Errorf("sqlCheckTrigramAvailable failed: %v", err)
|
|
}
|
|
|
|
// Verify summaries_fts SQL executes without error
|
|
_, err = db.Exec(sqlCreateSummariesFTS)
|
|
if err != nil {
|
|
t.Errorf("sqlCreateSummariesFTS failed: %v", err)
|
|
}
|
|
|
|
// Verify messages_fts SQL executes without error
|
|
_, err = db.Exec(sqlCreateMessagesFTS)
|
|
if err != nil {
|
|
t.Errorf("sqlCreateMessagesFTS failed: %v", err)
|
|
}
|
|
}
|