2018-06-10 08:44:53 +02:00
|
|
|
package base
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
2018-06-24 17:49:48 +02:00
|
|
|
"fmt"
|
2019-02-16 15:01:48 +01:00
|
|
|
|
2018-06-24 17:49:48 +02:00
|
|
|
"github.com/jouir/pgterminate/log"
|
2018-06-10 08:44:53 +02:00
|
|
|
"github.com/lib/pq"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
maxQueryLength = 1000
|
|
|
|
)
|
|
|
|
|
|
|
|
// Db centralizes connection to the database
|
|
|
|
type Db struct {
|
|
|
|
dsn string
|
|
|
|
conn *sql.DB
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewDb creates a Db object
|
|
|
|
func NewDb(dsn string) *Db {
|
|
|
|
return &Db{
|
|
|
|
dsn: dsn,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Connect connects to the instance and ping it to ensure connection is working
|
|
|
|
func (db *Db) Connect() {
|
|
|
|
conn, err := sql.Open("postgres", db.dsn)
|
|
|
|
Panic(err)
|
|
|
|
|
|
|
|
err = conn.Ping()
|
|
|
|
Panic(err)
|
|
|
|
|
|
|
|
db.conn = conn
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disconnect ends connection cleanly
|
|
|
|
func (db *Db) Disconnect() {
|
|
|
|
err := db.conn.Close()
|
|
|
|
Panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sessions connects to the database and returns current sessions
|
2018-07-08 23:48:48 +02:00
|
|
|
func (db *Db) Sessions() (sessions []*Session) {
|
2018-06-24 17:49:48 +02:00
|
|
|
query := fmt.Sprintf(`select pid as pid,
|
|
|
|
usename as user,
|
|
|
|
datname as db,
|
2018-07-08 19:44:24 +02:00
|
|
|
coalesce(host(client_addr)::text || ':' || client_port::text, 'localhost') as client,
|
2018-06-24 17:49:48 +02:00
|
|
|
state as state, substring(query from 1 for %d) as query,
|
2019-02-16 15:01:48 +01:00
|
|
|
coalesce(extract(epoch from now() - state_change), 0) as "stateDuration",
|
|
|
|
application_name as "applicationName"
|
2018-06-24 17:49:48 +02:00
|
|
|
from pg_catalog.pg_stat_activity
|
|
|
|
where pid <> pg_backend_pid();`, maxQueryLength)
|
|
|
|
log.Debugf("query: %s\n", query)
|
2018-06-10 08:44:53 +02:00
|
|
|
rows, err := db.conn.Query(query)
|
|
|
|
Panic(err)
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
for rows.Next() {
|
|
|
|
var pid sql.NullInt64
|
2019-02-16 15:01:48 +01:00
|
|
|
var user, db, client, state, query, applicationName sql.NullString
|
2018-06-11 21:02:24 +02:00
|
|
|
var stateDuration float64
|
2019-02-16 15:01:48 +01:00
|
|
|
err := rows.Scan(&pid, &user, &db, &client, &state, &query, &stateDuration, &applicationName)
|
2018-06-10 08:44:53 +02:00
|
|
|
Panic(err)
|
|
|
|
|
2019-02-16 15:01:48 +01:00
|
|
|
if pid.Valid && user.Valid && db.Valid && client.Valid && state.Valid && query.Valid && applicationName.Valid {
|
|
|
|
sessions = append(sessions, NewSession(pid.Int64, user.String, db.String, client.String, state.String, query.String, stateDuration, applicationName.String))
|
2018-06-10 08:44:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return sessions
|
|
|
|
}
|
|
|
|
|
|
|
|
// TerminateSessions terminates a list of sessions
|
2018-07-08 23:48:48 +02:00
|
|
|
func (db *Db) TerminateSessions(sessions []*Session) {
|
2018-06-10 08:44:53 +02:00
|
|
|
var pids []int64
|
|
|
|
for _, session := range sessions {
|
|
|
|
pids = append(pids, session.Pid)
|
|
|
|
}
|
|
|
|
if len(pids) > 0 {
|
|
|
|
query := `select pg_terminate_backend(pid) from pg_stat_activity where pid = any($1);`
|
2018-06-24 17:49:48 +02:00
|
|
|
log.Debugf("query: %s\n", query)
|
2018-06-10 08:44:53 +02:00
|
|
|
_, err := db.conn.Exec(query, pq.Array(pids))
|
|
|
|
Panic(err)
|
|
|
|
}
|
|
|
|
}
|
2018-06-30 11:11:24 +02:00
|
|
|
|
|
|
|
// CancelSessions terminates current query of a list of sessions
|
2018-07-08 23:48:48 +02:00
|
|
|
func (db *Db) CancelSessions(sessions []*Session) {
|
2018-06-30 11:11:24 +02:00
|
|
|
var pids []int64
|
|
|
|
for _, session := range sessions {
|
|
|
|
pids = append(pids, session.Pid)
|
|
|
|
}
|
|
|
|
if len(pids) > 0 {
|
|
|
|
query := `select pg_cancel_backend(pid) from pg_stat_activity where pid = any($1);`
|
|
|
|
log.Debugf("query: %s\n", query)
|
|
|
|
_, err := db.conn.Exec(query, pq.Array(pids))
|
|
|
|
Panic(err)
|
|
|
|
}
|
|
|
|
}
|