diff options
| author | Marko Tiikkaja <marko@joh.to> | 2015-03-27 19:45:12 +0100 |
|---|---|---|
| committer | Brad Fitzpatrick <bradfitz@golang.org> | 2015-04-08 16:18:36 +0000 |
| commit | c468f94672af25bc34975ba96309e20e972fa340 (patch) | |
| tree | 3647ad6bb35eb2969a7f4c0231b9c17e4434f058 /src/database/sql/sql.go | |
| parent | d5ef69814204f1bf8bee75bc89efcbcb33a64606 (diff) | |
| download | go-c468f94672af25bc34975ba96309e20e972fa340.tar.xz | |
database/sql: Retry with a fresh connection after maxBadConnRetries
Previously if the connection pool was larger than maxBadConnRetries
and there were a lot of bad connections in the pool (for example if
the database server was restarted), a query might have failed with an
ErrBadConn unnecessarily. Instead of trying to guess how many times
to retry, try maxBadConnRetries times and then force a fresh
connection to be used for the last attempt. At the same time, lower
maxBadConnRetries to a smaller value now that it's not that important
to retry so many times from the free connection list.
Fixes #8834
Change-Id: I6542f151a766a658980fb396fa4880ecf5874e3d
Reviewed-on: https://go-review.googlesource.com/2034
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
Diffstat (limited to 'src/database/sql/sql.go')
| -rw-r--r-- | src/database/sql/sql.go | 87 |
1 files changed, 57 insertions, 30 deletions
diff --git a/src/database/sql/sql.go b/src/database/sql/sql.go index 0a84163a03..96c93ed1c6 100644 --- a/src/database/sql/sql.go +++ b/src/database/sql/sql.go @@ -235,6 +235,18 @@ type DB struct { maxOpen int // <= 0 means unlimited } +// connReuseStrategy determines how (*DB).conn returns database connections. +type connReuseStrategy uint8 + +const ( + // alwaysNewConn forces a new connection to the database. + alwaysNewConn connReuseStrategy = iota + // cachedOrNewConn returns a cached connection, if available, else waits + // for one to become available (if MaxOpenConns has been reached) or + // creates a new database connection. + cachedOrNewConn +) + // driverConn wraps a driver.Conn with a mutex, to // be held during all calls into the Conn. (including any calls onto // interfaces returned via that Conn, such as calls on Tx, Stmt, @@ -465,7 +477,7 @@ func (db *DB) Ping() error { // TODO(bradfitz): give drivers an optional hook to implement // this in a more efficient or more reliable way, if they // have one. - dc, err := db.conn() + dc, err := db.conn(cachedOrNewConn) if err != nil { return err } @@ -651,17 +663,28 @@ type connRequest struct { var errDBClosed = errors.New("sql: database is closed") -// conn returns a newly-opened or cached *driverConn -func (db *DB) conn() (*driverConn, error) { +// conn returns a newly-opened or cached *driverConn. +func (db *DB) conn(strategy connReuseStrategy) (*driverConn, error) { db.mu.Lock() if db.closed { db.mu.Unlock() return nil, errDBClosed } - // If db.maxOpen > 0 and the number of open connections is over the limit - // and there are no free connection, make a request and wait. - if db.maxOpen > 0 && db.numOpen >= db.maxOpen && len(db.freeConn) == 0 { + // Prefer a free connection, if possible. + numFree := len(db.freeConn) + if strategy == cachedOrNewConn && numFree > 0 { + conn := db.freeConn[0] + copy(db.freeConn, db.freeConn[1:]) + db.freeConn = db.freeConn[:numFree-1] + conn.inUse = true + db.mu.Unlock() + return conn, nil + } + + // Out of free connections or we were asked not to use one. If we're not + // allowed to open any more connections, make a request and wait. + if db.maxOpen > 0 && db.numOpen >= db.maxOpen { // Make the connRequest channel. It's buffered so that the // connectionOpener doesn't block while waiting for the req to be read. req := make(chan connRequest, 1) @@ -671,15 +694,6 @@ func (db *DB) conn() (*driverConn, error) { return ret.conn, ret.err } - if c := len(db.freeConn); c > 0 { - conn := db.freeConn[0] - copy(db.freeConn, db.freeConn[1:]) - db.freeConn = db.freeConn[:c-1] - conn.inUse = true - db.mu.Unlock() - return conn, nil - } - db.numOpen++ // optimistically db.mu.Unlock() ci, err := db.driver.Open(db.dsn) @@ -808,8 +822,9 @@ func (db *DB) putConnDBLocked(dc *driverConn, err error) bool { } // maxBadConnRetries is the number of maximum retries if the driver returns -// driver.ErrBadConn to signal a broken connection. -const maxBadConnRetries = 10 +// driver.ErrBadConn to signal a broken connection before forcing a new +// connection to be opened. +const maxBadConnRetries = 2 // Prepare creates a prepared statement for later queries or executions. // Multiple queries or executions may be run concurrently from the @@ -818,22 +833,25 @@ func (db *DB) Prepare(query string) (*Stmt, error) { var stmt *Stmt var err error for i := 0; i < maxBadConnRetries; i++ { - stmt, err = db.prepare(query) + stmt, err = db.prepare(query, cachedOrNewConn) if err != driver.ErrBadConn { break } } + if err == driver.ErrBadConn { + return db.prepare(query, alwaysNewConn) + } return stmt, err } -func (db *DB) prepare(query string) (*Stmt, error) { +func (db *DB) prepare(query string, strategy connReuseStrategy) (*Stmt, error) { // TODO: check if db.driver supports an optional // driver.Preparer interface and call that instead, if so, // otherwise we make a prepared statement that's bound // to a connection, and to execute this prepared statement // we either need to use this connection (if it's free), else // get a new connection + re-prepare + execute on that one. - dc, err := db.conn() + dc, err := db.conn(strategy) if err != nil { return nil, err } @@ -861,16 +879,19 @@ func (db *DB) Exec(query string, args ...interface{}) (Result, error) { var res Result var err error for i := 0; i < maxBadConnRetries; i++ { - res, err = db.exec(query, args) + res, err = db.exec(query, args, cachedOrNewConn) if err != driver.ErrBadConn { break } } + if err == driver.ErrBadConn { + return db.exec(query, args, alwaysNewConn) + } return res, err } -func (db *DB) exec(query string, args []interface{}) (res Result, err error) { - dc, err := db.conn() +func (db *DB) exec(query string, args []interface{}, strategy connReuseStrategy) (res Result, err error) { + dc, err := db.conn(strategy) if err != nil { return nil, err } @@ -910,16 +931,19 @@ func (db *DB) Query(query string, args ...interface{}) (*Rows, error) { var rows *Rows var err error for i := 0; i < maxBadConnRetries; i++ { - rows, err = db.query(query, args) + rows, err = db.query(query, args, cachedOrNewConn) if err != driver.ErrBadConn { break } } + if err == driver.ErrBadConn { + return db.query(query, args, alwaysNewConn) + } return rows, err } -func (db *DB) query(query string, args []interface{}) (*Rows, error) { - ci, err := db.conn() +func (db *DB) query(query string, args []interface{}, strategy connReuseStrategy) (*Rows, error) { + ci, err := db.conn(strategy) if err != nil { return nil, err } @@ -998,16 +1022,19 @@ func (db *DB) Begin() (*Tx, error) { var tx *Tx var err error for i := 0; i < maxBadConnRetries; i++ { - tx, err = db.begin() + tx, err = db.begin(cachedOrNewConn) if err != driver.ErrBadConn { break } } + if err == driver.ErrBadConn { + return db.begin(alwaysNewConn) + } return tx, err } -func (db *DB) begin() (tx *Tx, err error) { - dc, err := db.conn() +func (db *DB) begin(strategy connReuseStrategy) (tx *Tx, err error) { + dc, err := db.conn(strategy) if err != nil { return nil, err } @@ -1396,7 +1423,7 @@ func (s *Stmt) connStmt() (ci *driverConn, releaseConn func(error), si driver.St s.mu.Unlock() // TODO(bradfitz): or always wait for one? make configurable later? - dc, err := s.db.conn() + dc, err := s.db.conn(cachedOrNewConn) if err != nil { return nil, nil, nil, err } |
