-
-
Notifications
You must be signed in to change notification settings - Fork 69
/
sql.go
105 lines (90 loc) · 2.7 KB
/
sql.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
package sql_exporter
import (
"context"
"database/sql"
"errors"
"fmt"
"log/slog"
"net/url"
"os"
"time"
"github.com/xo/dburl"
)
// OpenConnection parses a provided DSN, and opens a DB handle ensuring early termination if the context is closed
// (this is actually prevented by `database/sql` implementation), sets connection limits and returns the handle.
func OpenConnection(ctx context.Context, logContext, dsn string, maxConns, maxIdleConns int, maxConnLifetime time.Duration) (*sql.DB, error) {
var (
url *dburl.URL
conn *sql.DB
err error
ch = make(chan error)
)
url, err = safeParse(dsn)
if err != nil {
return nil, err
}
driver := url.Driver
if url.GoDriver != "" {
driver = url.GoDriver
}
// Open the DB handle in a separate goroutine so we can terminate early if the context closes.
go func() {
conn, err = sql.Open(driver, url.DSN)
close(ch)
}()
select {
case <-ctx.Done():
return nil, ctx.Err()
case <-ch:
if err != nil {
return nil, err
}
}
conn.SetMaxIdleConns(maxIdleConns)
conn.SetMaxOpenConns(maxConns)
conn.SetConnMaxLifetime(maxConnLifetime)
slog.Debug("Database handle successfully opened", "logContext", logContext, "driver", driver)
return conn, nil
}
// PingDB is a wrapper around sql.DB.PingContext() that terminates as soon as the context is closed.
//
// sql.DB does not actually pass along the context to the driver when opening a connection (which always happens if the
// database is down) and the driver uses an arbitrary timeout which may well be longer than ours. So we run the ping
// call in a goroutine and terminate immediately if the context is closed.
func PingDB(ctx context.Context, conn *sql.DB) error {
ch := make(chan error, 1)
go func() {
ch <- conn.PingContext(ctx)
close(ch)
}()
select {
case <-ctx.Done():
return ctx.Err()
case err := <-ch:
return err
}
}
// safeParse wraps dburl.Parse method in order to prevent leaking credentials
// if underlying url parse failed. By default it returns a raw url string in error message,
// which most likely contains a password. It's undesired here.
func safeParse(rawURL string) (*dburl.URL, error) {
parsed, err := dburl.Parse(expandEnv(rawURL))
if err != nil {
if uerr := new(url.Error); errors.As(err, &uerr) {
return nil, uerr.Err
}
return nil, errors.New("invalid URL")
}
return parsed, nil
}
// expandEnv falls back to the original env variable if not found for better readability
func expandEnv(env string) string {
lookupFunc := func(env string) string {
if value, ok := os.LookupEnv(env); ok {
return value
}
slog.Error("Environment variable is not found, cannot expand", "env", env)
return fmt.Sprintf("$%s", env)
}
return os.Expand(env, lookupFunc)
}