Files
TermorServer/internal/db/db.go
benya 35abd27473 feat: add sqlite-backed auth and library services
Bootstrap SQLite on server startup with embedded migrations and development seed data. Replace placeholder auth and library responses with database-backed services, bearer sessions, and repository-driven API handlers.
2026-04-02 22:22:38 +03:00

72 lines
1.5 KiB
Go

package db
import (
"context"
"database/sql"
"embed"
"fmt"
"os"
"path/filepath"
"sort"
"time"
_ "modernc.org/sqlite"
"github.com/benya/temporserv/internal/config"
)
//go:embed migrations/*.sql
var migrationFiles embed.FS
func Open(ctx context.Context, cfg config.Config) (*sql.DB, error) {
if err := os.MkdirAll(filepath.Dir(cfg.DatabasePath), 0o755); err != nil {
return nil, fmt.Errorf("create database directory: %w", err)
}
db, err := sql.Open("sqlite", cfg.DatabasePath)
if err != nil {
return nil, fmt.Errorf("open database: %w", err)
}
db.SetMaxOpenConns(1)
db.SetConnMaxLifetime(30 * time.Minute)
if err := db.PingContext(ctx); err != nil {
return nil, fmt.Errorf("ping database: %w", err)
}
if _, err := db.ExecContext(ctx, "PRAGMA foreign_keys = ON;"); err != nil {
return nil, fmt.Errorf("enable foreign keys: %w", err)
}
return db, nil
}
func Migrate(ctx context.Context, database *sql.DB) error {
entries, err := migrationFiles.ReadDir("migrations")
if err != nil {
return fmt.Errorf("read migrations: %w", err)
}
sort.Slice(entries, func(i, j int) bool {
return entries[i].Name() < entries[j].Name()
})
for _, entry := range entries {
if entry.IsDir() {
continue
}
sqlBytes, err := migrationFiles.ReadFile("migrations/" + entry.Name())
if err != nil {
return fmt.Errorf("read migration %s: %w", entry.Name(), err)
}
if _, err := database.ExecContext(ctx, string(sqlBytes)); err != nil {
return fmt.Errorf("apply migration %s: %w", entry.Name(), err)
}
}
return nil
}