Update ConnectionManager to still allow component defined connections (#3154)
This commit is contained in:
parent
9582827493
commit
e216c2fbf0
|
@ -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")
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue