package postgres import ( "context" "database/sql" "fmt" "sync" "time" _ "github.com/jackc/pgx/v5/stdlib" ) type Config struct { Host string `koanf:"host"` Port int `koanf:"port"` User string `koanf:"user"` Password string `koanf:"password"` DbName string `koanf:"dbName"` SSLMode string `koanf:"sslMode"` MaxIdleConn int `koanf:"maxIdleConns"` MaxOpenConn int `koanf:"maxOpenConns"` ConnMaxLifetime int `koanf:"connMaxLifetime"` } type DB struct { config Config db *sql.DB mu sync.Mutex statements map[statementKey]*sql.Stmt } func (db *DB) Conn() *sql.DB { return db.db } func New(config Config) *DB { dsn := fmt.Sprintf("host=%s port=%d user=%s password=%s dbname=%s sslmode=%s", config.Host, config.Port, config.User, config.Password, config.DbName, config.SSLMode) db, err := sql.Open("pgx", dsn) if err != nil { panic(fmt.Errorf("can't open postgres db: %w", err)) } maxIdle := config.MaxIdleConn maxOpen := config.MaxOpenConn lifetime := time.Duration(config.ConnMaxLifetime) * time.Second db.SetMaxIdleConns(maxIdle) db.SetMaxOpenConns(maxOpen) db.SetConnMaxLifetime(lifetime) return &DB{ config: config, db: db, statements: make(map[statementKey]*sql.Stmt), } } func (db *DB) PrepareStatement(ctx context.Context, key statementKey, query string) (*sql.Stmt, error) { db.mu.Lock() defer db.mu.Unlock() if stmt, ok := db.statements[key]; ok { return stmt, nil } stmt, err := db.db.PrepareContext(ctx, query) if err != nil { return nil, fmt.Errorf("prepare statement %q: %w", key, err) } db.statements[key] = stmt return stmt, nil } func (db *DB) CloseStatements() error { db.mu.Lock() defer db.mu.Unlock() var lastErr error for key, stmt := range db.statements { if err := stmt.Close(); err != nil { lastErr = err } delete(db.statements, key) } return lastErr } func (db *DB) Close() error { return db.db.Close() } func (db *DB) StmtQueryContext(ctx context.Context, stmt *sql.Stmt, args ...any) (*sql.Rows, error) { return stmt.QueryContext(ctx, args...) } func (db *DB) StmtQueryRowContext(ctx context.Context, stmt *sql.Stmt, args ...any) *sql.Row { return stmt.QueryRowContext(ctx, args...) } func (db *DB) StmtExecContext(ctx context.Context, stmt *sql.Stmt, args ...any) (sql.Result, error) { result, err := stmt.ExecContext(ctx, args...) if err != nil { return nil, err } return result, nil }