Expose db connection max lifetime and idle time settings

This commit is contained in:
Tulir Asokan 2022-02-23 14:30:21 +02:00
parent 77bef42ace
commit dbe25bbce6
5 changed files with 48 additions and 20 deletions

View file

@ -1,5 +1,5 @@
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
// Copyright (C) 2019 Tulir Asokan // Copyright (C) 2022 Tulir Asokan
// //
// This program is free software: you can redistribute it and/or modify // This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by // it under the terms of the GNU Affero General Public License as published by
@ -41,13 +41,7 @@ type Config struct {
Hostname string `yaml:"hostname"` Hostname string `yaml:"hostname"`
Port uint16 `yaml:"port"` Port uint16 `yaml:"port"`
Database struct { Database DatabaseConfig `yaml:"database"`
Type string `yaml:"type"`
URI string `yaml:"uri"`
MaxOpenConns int `yaml:"max_open_conns"`
MaxIdleConns int `yaml:"max_idle_conns"`
} `yaml:"database"`
Provisioning struct { Provisioning struct {
Prefix string `yaml:"prefix"` Prefix string `yaml:"prefix"`
@ -131,3 +125,14 @@ func (config *Config) MakeAppService() (*appservice.AppService, error) {
as.Registration, err = config.GetRegistration() as.Registration, err = config.GetRegistration()
return as, err return as, err
} }
type DatabaseConfig struct {
Type string `yaml:"type"`
URI string `yaml:"uri"`
MaxOpenConns int `yaml:"max_open_conns"`
MaxIdleConns int `yaml:"max_idle_conns"`
ConnMaxIdleTime string `yaml:"conn_max_idle_time"`
ConnMaxLifetime string `yaml:"conn_max_lifetime"`
}

View file

@ -1,5 +1,5 @@
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
// Copyright (C) 2021 Tulir Asokan // Copyright (C) 2022 Tulir Asokan
// //
// This program is free software: you can redistribute it and/or modify // This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by // it under the terms of the GNU Affero General Public License as published by
@ -40,6 +40,8 @@ func (helper *UpgradeHelper) doUpgrade() {
helper.Copy(Str, "appservice", "database", "uri") helper.Copy(Str, "appservice", "database", "uri")
helper.Copy(Int, "appservice", "database", "max_open_conns") helper.Copy(Int, "appservice", "database", "max_open_conns")
helper.Copy(Int, "appservice", "database", "max_idle_conns") helper.Copy(Int, "appservice", "database", "max_idle_conns")
helper.Copy(Str|Null, "appservice", "database", "max_conn_idle_time")
helper.Copy(Str|Null, "appservice", "database", "max_conn_lifetime")
helper.Copy(Str, "appservice", "provisioning", "prefix") helper.Copy(Str, "appservice", "provisioning", "prefix")
if secret, ok := helper.Get(Str, "appservice", "provisioning", "shared_secret"); !ok || secret == "generate" { if secret, ok := helper.Get(Str, "appservice", "provisioning", "shared_secret"); !ok || secret == "generate" {
sharedSecret := appservice.RandomString(64) sharedSecret := appservice.RandomString(64)

View file

@ -1,5 +1,5 @@
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
// Copyright (C) 2019 Tulir Asokan // Copyright (C) 2022 Tulir Asokan
// //
// This program is free software: you can redistribute it and/or modify // This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by // it under the terms of the GNU Affero General Public License as published by
@ -18,6 +18,8 @@ package database
import ( import (
"database/sql" "database/sql"
"fmt"
"time"
"github.com/lib/pq" "github.com/lib/pq"
_ "github.com/mattn/go-sqlite3" _ "github.com/mattn/go-sqlite3"
@ -25,6 +27,7 @@ import (
log "maunium.net/go/maulogger/v2" log "maunium.net/go/maulogger/v2"
"go.mau.fi/whatsmeow/store/sqlstore" "go.mau.fi/whatsmeow/store/sqlstore"
"maunium.net/go/mautrix-whatsapp/config"
"maunium.net/go/mautrix-whatsapp/database/upgrades" "maunium.net/go/mautrix-whatsapp/database/upgrades"
) )
@ -45,8 +48,8 @@ type Database struct {
DisappearingMessage *DisappearingMessageQuery DisappearingMessage *DisappearingMessageQuery
} }
func New(dbType string, uri string, baseLog log.Logger) (*Database, error) { func New(cfg config.DatabaseConfig, baseLog log.Logger) (*Database, error) {
conn, err := sql.Open(dbType, uri) conn, err := sql.Open(cfg.Type, cfg.URI)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -54,7 +57,7 @@ func New(dbType string, uri string, baseLog log.Logger) (*Database, error) {
db := &Database{ db := &Database{
DB: conn, DB: conn,
log: baseLog.Sub("Database"), log: baseLog.Sub("Database"),
dialect: dbType, dialect: cfg.Type,
} }
db.User = &UserQuery{ db.User = &UserQuery{
db: db, db: db,
@ -76,6 +79,23 @@ func New(dbType string, uri string, baseLog log.Logger) (*Database, error) {
db: db, db: db,
log: db.log.Sub("DisappearingMessage"), log: db.log.Sub("DisappearingMessage"),
} }
db.SetMaxOpenConns(cfg.MaxOpenConns)
db.SetMaxIdleConns(cfg.MaxIdleConns)
if len(cfg.ConnMaxIdleTime) > 0 {
maxIdleTimeDuration, err := time.ParseDuration(cfg.ConnMaxIdleTime)
if err != nil {
return nil, fmt.Errorf("failed to parse max_conn_idle_time: %w", err)
}
db.SetConnMaxIdleTime(maxIdleTimeDuration)
}
if len(cfg.ConnMaxLifetime) > 0 {
maxLifetimeDuration, err := time.ParseDuration(cfg.ConnMaxLifetime)
if err != nil {
return nil, fmt.Errorf("failed to parse max_conn_idle_time: %w", err)
}
db.SetConnMaxLifetime(maxLifetimeDuration)
}
return db, nil return db, nil
} }

View file

@ -36,6 +36,10 @@ appservice:
# Maximum number of connections. Mostly relevant for Postgres. # Maximum number of connections. Mostly relevant for Postgres.
max_open_conns: 20 max_open_conns: 20
max_idle_conns: 2 max_idle_conns: 2
# Maximum connection idle time and lifetime before they're closed. Disabled if null.
# Parsed with https://pkg.go.dev/time#ParseDuration
max_conn_idle_time: null
max_conn_lifetime: null
# Settings for provisioning API # Settings for provisioning API
provisioning: provisioning:

11
main.go
View file

@ -1,5 +1,5 @@
// mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
// Copyright (C) 2021 Tulir Asokan // Copyright (C) 2022 Tulir Asokan
// //
// This program is free software: you can redistribute it and/or modify // This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by // it under the terms of the GNU Affero General Public License as published by
@ -136,7 +136,7 @@ func (bridge *Bridge) GenerateRegistration() {
} }
func (bridge *Bridge) MigrateDatabase() { func (bridge *Bridge) MigrateDatabase() {
oldDB, err := database.New(flag.Arg(0), flag.Arg(1), log.DefaultLogger) oldDB, err := database.New(config.DatabaseConfig{Type: flag.Arg(0), URI: flag.Arg(1)}, log.DefaultLogger)
if err != nil { if err != nil {
fmt.Println("Failed to open old database:", err) fmt.Println("Failed to open old database:", err)
os.Exit(30) os.Exit(30)
@ -147,7 +147,7 @@ func (bridge *Bridge) MigrateDatabase() {
os.Exit(31) os.Exit(31)
} }
newDB, err := database.New(bridge.Config.AppService.Database.Type, bridge.Config.AppService.Database.URI, log.DefaultLogger) newDB, err := database.New(bridge.Config.AppService.Database, log.DefaultLogger)
if err != nil { if err != nil {
fmt.Println("Failed to open new database:", err) fmt.Println("Failed to open new database:", err)
os.Exit(32) os.Exit(32)
@ -250,7 +250,7 @@ func (bridge *Bridge) Init() {
bridge.Log.Infoln("Initializing", VersionString) bridge.Log.Infoln("Initializing", VersionString)
bridge.Log.Debugln("Initializing database connection") bridge.Log.Debugln("Initializing database connection")
bridge.DB, err = database.New(bridge.Config.AppService.Database.Type, bridge.Config.AppService.Database.URI, bridge.Log) bridge.DB, err = database.New(bridge.Config.AppService.Database, bridge.Log)
if err != nil { if err != nil {
bridge.Log.Fatalln("Failed to initialize database connection:", err) bridge.Log.Fatalln("Failed to initialize database connection:", err)
os.Exit(14) os.Exit(14)
@ -260,9 +260,6 @@ func (bridge *Bridge) Init() {
bridge.StateStore = database.NewSQLStateStore(bridge.DB) bridge.StateStore = database.NewSQLStateStore(bridge.DB)
bridge.AS.StateStore = bridge.StateStore bridge.AS.StateStore = bridge.StateStore
bridge.DB.SetMaxOpenConns(bridge.Config.AppService.Database.MaxOpenConns)
bridge.DB.SetMaxIdleConns(bridge.Config.AppService.Database.MaxIdleConns)
bridge.WAContainer = sqlstore.NewWithDB(bridge.DB.DB, bridge.Config.AppService.Database.Type, nil) bridge.WAContainer = sqlstore.NewWithDB(bridge.DB.DB, bridge.Config.AppService.Database.Type, nil)
ss := bridge.Config.AppService.Provisioning.SharedSecret ss := bridge.Config.AppService.Provisioning.SharedSecret