Add cancel option to terminate queries

This commit is contained in:
Julien Riou 2018-06-30 11:11:24 +02:00
parent 750db87046
commit dcb07c225e
No known key found for this signature in database
GPG key ID: BA3E15176E45E85D
5 changed files with 22 additions and 1 deletions

View file

@ -8,6 +8,7 @@ With `pgterminate`, you shouldn't be paged at night because some queries has loc
# Highlights # Highlights
* `pgterminate` name is derived from `pg_terminate_backend` function, it terminates backends. * `pgterminate` name is derived from `pg_terminate_backend` function, it terminates backends.
* backends are called sessions in `pgterminate`. * backends are called sessions in `pgterminate`.
* use `cancel` option to terminate query instead of session.
* `active` sessions are backends in `active` state for more than `active-timeout` seconds. * `active` sessions are backends in `active` state for more than `active-timeout` seconds.
* `idle` sessions are backends in `idle`, `idle in transaction` or `idle in transaction (abort)` state for more than `idle-timeout` seconds. * `idle` sessions are backends in `idle`, `idle in transaction` or `idle in transaction (abort)` state for more than `idle-timeout` seconds.
* at least one of `active-timeout` and `idle-timeout` parameter is required, both can be used. * at least one of `active-timeout` and `idle-timeout` parameter is required, both can be used.

View file

@ -38,6 +38,7 @@ type Config struct {
ExcludeUsers StringFlags `yaml:"exclude-users"` ExcludeUsers StringFlags `yaml:"exclude-users"`
ExcludeUsersRegex string `yaml:"exclude-users-regex"` ExcludeUsersRegex string `yaml:"exclude-users-regex"`
ExcludeUsersRegexCompiled *regexp.Regexp ExcludeUsersRegexCompiled *regexp.Regexp
Cancel bool `yaml:"cancel"`
} }
func init() { func init() {

View file

@ -84,3 +84,17 @@ func (db *Db) TerminateSessions(sessions []Session) {
Panic(err) Panic(err)
} }
} }
// CancelSessions terminates current query of a list of sessions
func (db *Db) CancelSessions(sessions []Session) {
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)
}
}

View file

@ -47,6 +47,7 @@ func main() {
flag.StringVar(&config.IncludeUsersRegex, "include-users-regex", "", "Terminate users matching this regexp") flag.StringVar(&config.IncludeUsersRegex, "include-users-regex", "", "Terminate users matching this regexp")
flag.Var(&config.ExcludeUsers, "exclude-user", "Ignore this user (can be called multiple times)") flag.Var(&config.ExcludeUsers, "exclude-user", "Ignore this user (can be called multiple times)")
flag.StringVar(&config.ExcludeUsersRegex, "exclude-users-regex", "", "Ignore users matching this regexp") flag.StringVar(&config.ExcludeUsersRegex, "exclude-users-regex", "", "Ignore users matching this regexp")
flag.BoolVar(&config.Cancel, "cancel", false, "Cancel sessions instead of terminate")
flag.Parse() flag.Parse()
log.SetLevel(log.WarnLevel) log.SetLevel(log.WarnLevel)

View file

@ -55,7 +55,11 @@ func (t *Terminator) Run() {
// terminateAndNotify terminates a list of sessions and notifies channel // terminateAndNotify terminates a list of sessions and notifies channel
func (t *Terminator) terminateAndNotify(sessions []base.Session) { func (t *Terminator) terminateAndNotify(sessions []base.Session) {
t.db.TerminateSessions(sessions) if t.config.Cancel {
t.db.CancelSessions(sessions)
} else {
t.db.TerminateSessions(sessions)
}
for _, session := range sessions { for _, session := range sessions {
t.sessions <- session t.sessions <- session
} }