diff options
Diffstat (limited to 'backend/db.go')
| -rw-r--r-- | backend/db.go | 73 |
1 files changed, 73 insertions, 0 deletions
diff --git a/backend/db.go b/backend/db.go new file mode 100644 index 0000000..3559b9c --- /dev/null +++ b/backend/db.go @@ -0,0 +1,73 @@ +package backend + +import ( + "context" + "database/sql" + "embed" + "fmt" + "log/slog" + "regexp" + "slices" + "strconv" + + _ "github.com/mattn/go-sqlite3" +) + +//go:embed migrations +var migrations embed.FS + +var nameReg = regexp.MustCompile(`(\d{3})_[a-zA-Z_-]+.sql`) + +func ConnectDatabase(cfg *Config) *sql.DB { + db, err := sql.Open("sqlite3", fmt.Sprintf("file:%s?cache=shared", cfg.Database)) + if err != nil { + panic(err) + } + db.SetMaxOpenConns(1) + return db +} + +func RunMigration(ctx context.Context, db *sql.DB) error { + entries, err := migrations.ReadDir("migrations") + if err != nil { + return err + } + type dbConfig struct { + Id int + Migration int + } + type runMig struct { + val string + n int + } + var toRun []runMig + for _, e := range entries { + rawId := nameReg.FindStringSubmatch(e.Name()) + id, err := strconv.Atoi(rawId[1]) + if err != nil { + return err + } + b, err := migrations.ReadFile("migrations/" + e.Name()) + if err != nil { + return err + } + slog.Debug("loading migration", "n", id, "file", e.Name(), "content", string(b)) + toRun = append(toRun, runMig{ + val: string(b), n: id, + }) + } + if len(toRun) == 0 { + return nil + } + slices.SortFunc(toRun, func(a, b runMig) int { + return a.n - b.n + }) + for _, m := range toRun { + slog.Info("migrating", "n", m.n) + _, err := db.ExecContext(ctx, m.val) + if err != nil { + return err + } + } + return nil +} |
