connect.go 1.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364
  1. package db
  2. import (
  3. "context"
  4. "database/sql"
  5. "fmt"
  6. "log/slog"
  7. "path/filepath"
  8. "github.com/ncruces/go-sqlite3"
  9. "github.com/ncruces/go-sqlite3/driver"
  10. _ "github.com/ncruces/go-sqlite3/embed"
  11. "github.com/pressly/goose/v3"
  12. )
  13. func Connect(ctx context.Context, dataDir string) (*sql.DB, error) {
  14. if dataDir == "" {
  15. return nil, fmt.Errorf("data.dir is not set")
  16. }
  17. dbPath := filepath.Join(dataDir, "crush.db")
  18. // Set pragmas for better performance
  19. pragmas := []string{
  20. "PRAGMA foreign_keys = ON;",
  21. "PRAGMA journal_mode = WAL;",
  22. "PRAGMA page_size = 4096;",
  23. "PRAGMA cache_size = -8000;",
  24. "PRAGMA synchronous = NORMAL;",
  25. "PRAGMA secure_delete = ON;",
  26. }
  27. db, err := driver.Open(dbPath, func(c *sqlite3.Conn) error {
  28. for _, pragma := range pragmas {
  29. if err := c.Exec(pragma); err != nil {
  30. return fmt.Errorf("failed to set pragma `%s`: %w", pragma, err)
  31. }
  32. }
  33. return nil
  34. })
  35. if err != nil {
  36. return nil, fmt.Errorf("failed to open database: %w", err)
  37. }
  38. // Verify connection
  39. if err = db.PingContext(ctx); err != nil {
  40. db.Close()
  41. return nil, fmt.Errorf("failed to connect to database: %w", err)
  42. }
  43. goose.SetBaseFS(FS)
  44. if err := goose.SetDialect("sqlite3"); err != nil {
  45. slog.Error("Failed to set dialect", "error", err)
  46. return nil, fmt.Errorf("failed to set dialect: %w", err)
  47. }
  48. if err := goose.Up(db, "migrations"); err != nil {
  49. slog.Error("Failed to apply migrations", "error", err)
  50. return nil, fmt.Errorf("failed to apply migrations: %w", err)
  51. }
  52. return db, nil
  53. }