Update ConnectionManager to still allow component defined connections (#3154)

This commit is contained in:
Till 2023-07-21 08:34:01 +02:00 committed by GitHub
parent 9582827493
commit e216c2fbf0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 61 additions and 30 deletions

View file

@ -17,16 +17,21 @@ package sqlutil
import ( import (
"database/sql" "database/sql"
"fmt" "fmt"
"sync"
"github.com/matrix-org/dendrite/setup/config" "github.com/matrix-org/dendrite/setup/config"
"github.com/matrix-org/dendrite/setup/process" "github.com/matrix-org/dendrite/setup/process"
) )
type Connections struct { type Connections struct {
db *sql.DB globalConfig config.DatabaseOptions
writer Writer processContext *process.ProcessContext
globalConfig config.DatabaseOptions existingConnections sync.Map
processContext *process.ProcessContext }
type con struct {
db *sql.DB
writer Writer
} }
func NewConnectionManager(processCtx *process.ProcessContext, globalConfig config.DatabaseOptions) *Connections { func NewConnectionManager(processCtx *process.ProcessContext, globalConfig config.DatabaseOptions) *Connections {
@ -38,9 +43,13 @@ func NewConnectionManager(processCtx *process.ProcessContext, globalConfig confi
func (c *Connections) Connection(dbProperties *config.DatabaseOptions) (*sql.DB, Writer, error) { func (c *Connections) Connection(dbProperties *config.DatabaseOptions) (*sql.DB, Writer, error) {
var err error var err error
// If no connectionString was provided, try the global one
if dbProperties.ConnectionString == "" { if dbProperties.ConnectionString == "" {
// if no connectionString was provided, try the global one
dbProperties = &c.globalConfig dbProperties = &c.globalConfig
// If we still don't have a connection string, that's a problem
if dbProperties.ConnectionString == "" {
return nil, nil, fmt.Errorf("no database connections configured")
}
} }
writer := NewDummyWriter() writer := NewDummyWriter()
@ -48,30 +57,30 @@ func (c *Connections) Connection(dbProperties *config.DatabaseOptions) (*sql.DB,
writer = NewExclusiveWriter() writer = NewExclusiveWriter()
} }
if dbProperties.ConnectionString != "" && c.db == nil { existing, loaded := c.existingConnections.LoadOrStore(dbProperties.ConnectionString, &con{})
// Open a new database connection using the supplied config. if loaded {
c.db, err = Open(dbProperties, writer) // We found an existing connection
if err != nil { ex := existing.(*con)
return nil, nil, err return ex.db, ex.writer, nil
}
// Open a new database connection using the supplied config.
db, err := Open(dbProperties, writer)
if err != nil {
return nil, nil, err
}
c.existingConnections.Store(dbProperties.ConnectionString, &con{db: db, writer: writer})
go func() {
if c.processContext == nil {
return
} }
c.writer = writer // If we have a ProcessContext, start a component and wait for
go func() { // Dendrite to shut down to cleanly close the database connection.
if c.processContext == nil { c.processContext.ComponentStarted()
return <-c.processContext.WaitForShutdown()
} _ = db.Close()
// If we have a ProcessContext, start a component and wait for c.processContext.ComponentFinished()
// Dendrite to shut down to cleanly close the database connection. }()
c.processContext.ComponentStarted() return db, writer, nil
<-c.processContext.WaitForShutdown()
_ = c.db.Close()
c.processContext.ComponentFinished()
}()
return c.db, c.writer, nil
}
if c.db != nil && c.writer != nil {
// Ignore the supplied config and return the global pool and
// writer.
return c.db, c.writer, nil
}
return nil, nil, fmt.Errorf("no database connections configured")
} }

View file

@ -48,6 +48,22 @@ func TestConnectionManager(t *testing.T) {
if !reflect.DeepEqual(writer, writer2) { if !reflect.DeepEqual(writer, writer2) {
t.Fatalf("expected database writer to be reused") t.Fatalf("expected database writer to be reused")
} }
// This test does not work with Postgres, because we can't just simply append
// "x" or replace the database to use.
if dbType == test.DBTypePostgres {
return
}
// Test different connection string
dbProps = &config.DatabaseOptions{ConnectionString: config.DataSource(conStr + "x")}
db3, _, err := cm.Connection(dbProps)
if err != nil {
t.Fatal(err)
}
if reflect.DeepEqual(db, db3) {
t.Fatalf("expected different database connection")
}
}) })
}) })
@ -115,4 +131,10 @@ func TestConnectionManager(t *testing.T) {
if err == nil { if err == nil {
t.Fatal("expected an error but got none") t.Fatal("expected an error but got none")
} }
// empty connection string is not allowed
_, _, err = cm2.Connection(&config.DatabaseOptions{})
if err == nil {
t.Fatal("expected an error but got none")
}
} }