fixed "unable to open database file" (potentially), and errors wrapped in pkg/persistence

This commit is contained in:
Bora M. Alper 2018-12-24 18:27:36 +03:00
parent d5a03feb43
commit 64543d71d7
No known key found for this signature in database
GPG Key ID: 8F1A9504E1BD114D
2 changed files with 36 additions and 15 deletions

View File

@ -4,6 +4,7 @@ import (
"encoding/hex"
"encoding/json"
"fmt"
"github.com/pkg/errors"
"net/url"
"go.uber.org/zap"
@ -104,7 +105,7 @@ func MakeDatabase(rawURL string, logger *zap.Logger) (Database, error) {
url_, err := url.Parse(rawURL)
if err != nil {
return nil, err
return nil, errors.Wrap(err, "url.Parse")
}
switch url_.Scheme {

View File

@ -27,23 +27,43 @@ func makeSqlite3Database(url_ *url.URL) (Database, error) {
dbDir, _ := path.Split(url_.Path)
if err := os.MkdirAll(dbDir, 0755); err != nil {
return nil, fmt.Errorf("for directory `%s`: %s", dbDir, err.Error())
return nil, errors.Wrapf(err, "mkdirAll error for `%s`", dbDir)
}
var err error
url_.Scheme = ""
db.conn, err = sql.Open("sqlite3", url_.String())
if err != nil {
return nil, fmt.Errorf("sql.Open: %s", err.Error())
return nil, errors.Wrap(err, "sql.Open")
}
// > Open may just validate its arguments without creating a connection to the database. To
// > verify that the data source Name is valid, call Ping.
// https://golang.org/pkg/database/sql/#Open
if err = db.conn.Ping(); err != nil {
return nil, fmt.Errorf("sql.DB.Ping: %s", err.Error())
return nil, errors.Wrap(err, "sql.DB.Ping")
}
// > After some time we receive "unable to open database file" error while trying to execute a transaction using
// > Tx.Exec().
// -- boramalper
//
// > Not sure if this would be contributing to your issue, but one of the problems we've observed in the past is the
// > standard library's attempt to pool connections. (This makes more sense for database connections that are actual
// > network connections, as opposed to SQLite.)
// > Basically, the problem we encountered was that most pragmas (except specifically PRAGMA journal_mode=WAL, as
// > per the documentation) apply to the connection, so if the standard library is opening/closing connections
// > behind your back for pooling purposes, it can lead to unintended behavior.
// -- rittneje
//
// https://github.com/mattn/go-sqlite3/issues/618
//
// Our solution is to set the connection max lifetime to infinity (reuse connection forever), and max open
// connections to 1.
db.conn.SetConnMaxLifetime(0) // https://golang.org/pkg/database/sql/#DB.SetConnMaxLifetime
db.conn.SetMaxOpenConns(1)
db.conn.SetMaxIdleConns(0)
if err := db.setupDatabase(); err != nil {
return nil, errors.Wrap(err, "setupDatabase")
}
@ -299,7 +319,7 @@ func (db *sqlite3Database) QueryTorrents(
rows, err := db.conn.Query(sqlQuery, queryArgs...)
defer rows.Close()
if err != nil {
return nil, fmt.Errorf("error while querying torrents: %s", err.Error())
return nil, errors.Wrap(err, "query error")
}
torrents := make([]TorrentMetadata, 0)
@ -395,7 +415,7 @@ func (db *sqlite3Database) GetFiles(infoHash []byte) ([]File, error) {
func (db *sqlite3Database) GetStatistics(from string, n uint) (*Statistics, error) {
fromTime, gran, err := ParseISO8601(from)
if err != nil {
return nil, fmt.Errorf("error while parsing from: %s", err.Error())
return nil, errors.Wrap(err, "parsing ISO8601 error")
}
var toTime time.Time
@ -479,12 +499,12 @@ func (db *sqlite3Database) setupDatabase() error {
PRAGMA encoding='UTF-8';
`)
if err != nil {
return fmt.Errorf("sql.DB.Exec (PRAGMAs): %s", err.Error())
return errors.Wrap(err, "sql.DB.Exec (PRAGMAs)")
}
tx, err := db.conn.Begin()
if err != nil {
return fmt.Errorf("sql.DB.Begin: %s", err.Error())
return errors.Wrap(err, "sql.DB.Begin")
}
// If everything goes as planned and no error occurs, we will commit the transaction before
// returning from the function so the tx.Rollback() call will fail, trying to rollback a
@ -512,13 +532,13 @@ func (db *sqlite3Database) setupDatabase() error {
);
`)
if err != nil {
return fmt.Errorf("sql.Tx.Exec (v0): %s", err.Error())
return errors.Wrap(err, "sql.Tx.Exec (v0)")
}
// Get the user_version:
rows, err := tx.Query("PRAGMA user_version;")
if err != nil {
return fmt.Errorf("sql.Tx.Query (user_version): %s", err.Error())
return errors.Wrap(err, "sql.Tx.Query (user_version)")
}
defer rows.Close()
var userVersion int
@ -526,7 +546,7 @@ func (db *sqlite3Database) setupDatabase() error {
return fmt.Errorf("sql.Rows.Next (user_version): PRAGMA user_version did not return any rows!")
}
if err = rows.Scan(&userVersion); err != nil {
return fmt.Errorf("sql.Rows.Scan (user_version): %s", err.Error())
return errors.Wrap(err, "sql.Rows.Scan (user_version)")
}
switch userVersion {
@ -541,7 +561,7 @@ func (db *sqlite3Database) setupDatabase() error {
PRAGMA user_version = 1;
`)
if err != nil {
return fmt.Errorf("sql.Tx.Exec (v0 -> v1): %s", err.Error())
return errors.Wrap(err, "sql.Tx.Exec (v0 -> v1)")
}
fallthrough
@ -586,7 +606,7 @@ func (db *sqlite3Database) setupDatabase() error {
PRAGMA user_version = 2;
`)
if err != nil {
return fmt.Errorf("sql.Tx.Exec (v1 -> v2): %s", err.Error())
return errors.Wrap(err, "sql.Tx.Exec (v1 -> v2)")
}
fallthrough
@ -650,7 +670,7 @@ func (db *sqlite3Database) setupDatabase() error {
}
if err = tx.Commit(); err != nil {
return fmt.Errorf("sql.Tx.Commit: %s", err.Error())
return errors.Wrap(err, "sql.Tx.Commit")
}
return nil