mirror of
https://github.com/Dvorinka/Primora.git
synced 2026-06-04 04:23:00 +00:00
85 lines
2.0 KiB
Go
85 lines
2.0 KiB
Go
package database
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"fmt"
|
|
"log/slog"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/jackc/pgx/v5/pgxpool"
|
|
"github.com/jackc/pgx/v5/stdlib"
|
|
"github.com/pressly/goose/v3"
|
|
)
|
|
|
|
func Connect(ctx context.Context, databaseURL string) (*pgxpool.Pool, error) {
|
|
var lastErr error
|
|
for attempt := 1; attempt <= 20; attempt++ {
|
|
config, err := pgxpool.ParseConfig(databaseURL)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("parse database config: %w", err)
|
|
}
|
|
|
|
config.MaxConnLifetime = 30 * time.Minute
|
|
config.MaxConns = 20
|
|
config.MinConns = 2
|
|
|
|
pool, err := pgxpool.NewWithConfig(ctx, config)
|
|
if err != nil {
|
|
lastErr = fmt.Errorf("create pool: %w", err)
|
|
} else if err := pool.Ping(ctx); err != nil {
|
|
pool.Close()
|
|
lastErr = fmt.Errorf("ping database: %w", err)
|
|
} else {
|
|
return pool, nil
|
|
}
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil, ctx.Err()
|
|
case <-time.After(2 * time.Second):
|
|
}
|
|
}
|
|
return nil, lastErr
|
|
}
|
|
|
|
func RunMigrations(databaseURL, migrationsDir string, logger *slog.Logger) error {
|
|
if err := goose.SetDialect("postgres"); err != nil {
|
|
return fmt.Errorf("set goose dialect: %w", err)
|
|
}
|
|
sqlDB, err := sql.Open("pgx", databaseURL)
|
|
if err != nil {
|
|
return fmt.Errorf("open sql db: %w", err)
|
|
}
|
|
defer sqlDB.Close()
|
|
|
|
if err := goose.Up(sqlDB, migrationsDir); err != nil {
|
|
return fmt.Errorf("goose up: %w", err)
|
|
}
|
|
logger.Info("database migrations complete", "dir", migrationsDir)
|
|
return nil
|
|
}
|
|
|
|
func RunMigrationsFromPool(pool *pgxpool.Pool, migrationsDir string, logger *slog.Logger) error {
|
|
sqlDB := stdlib.OpenDBFromPool(pool)
|
|
defer sqlDB.Close()
|
|
|
|
if err := goose.SetDialect("postgres"); err != nil {
|
|
return fmt.Errorf("set goose dialect: %w", err)
|
|
}
|
|
if err := goose.Up(sqlDB, migrationsDir); err != nil {
|
|
return fmt.Errorf("goose up: %w", err)
|
|
}
|
|
logger.Info("database migrations complete", "dir", migrationsDir)
|
|
return nil
|
|
}
|
|
|
|
func ResolveMigrationsDir() string {
|
|
if dir := os.Getenv("BACKEND_MIGRATIONS_DIR"); dir != "" {
|
|
return dir
|
|
}
|
|
return filepath.Join("db", "migrations")
|
|
}
|