influxdb/sqlite/sqlite.go

380 lines
9.6 KiB
Go

package sqlite
import (
"context"
"database/sql"
"fmt"
"io"
"os"
"path/filepath"
"sync"
errors2 "github.com/influxdata/influxdb/v2/pkg/errors"
"github.com/influxdata/influxdb/v2/kit/tracing"
"github.com/influxdata/influxdb/v2/pkg/fs"
sqliteMigrations "github.com/influxdata/influxdb/v2/sqlite/migrations"
"github.com/jmoiron/sqlx"
"github.com/mattn/go-sqlite3"
"go.uber.org/zap"
)
const (
DefaultFilename = "influxd.sqlite"
InmemPath = ":memory:"
migrationsTableName = "migrations"
)
// SqlStore is a wrapper around the db and provides basic functionality for maintaining the db
// including flushing the data from the db during end-to-end testing.
type SqlStore struct {
Mu sync.RWMutex
DB *sqlx.DB
log *zap.Logger
path string
}
func NewSqlStore(path string, log *zap.Logger) (*SqlStore, error) {
s := &SqlStore{
log: log,
path: path,
}
if err := s.openDB(); err != nil {
return nil, err
}
return s, nil
}
// open the file at the specified path
func (s *SqlStore) openDB() error {
db, err := sqlx.Open("sqlite3", s.path)
if err != nil {
return err
}
s.log.Info("Resources opened", zap.String("path", s.path))
// If using an in-memory database, don't allow more than 1 connection. Each connection
// is given a "new" database. We can't use a shared cache in-memory database because
// parallel tests that run multiple launchers in the same process will have issues doing
// concurrent writes to the database. See: https://sqlite.org/inmemorydb.html
if s.path == InmemPath {
db.SetMaxOpenConns(1)
}
s.DB = db
return nil
}
// Close the connection to the sqlite database
func (s *SqlStore) Close() error {
err := s.DB.Close()
if err != nil {
return err
}
return nil
}
// RLockSqlStore locks the database using the mutex. This is intended to lock the database for writes.
// It is the responsibilty of implementing service code to manage locks for write operations.
func (s *SqlStore) RLockSqlStore() {
s.Mu.RLock()
}
// RUnlockSqlStore unlocks the database.
func (s *SqlStore) RUnlockSqlStore() {
s.Mu.RUnlock()
}
// Flush deletes all records for all tables in the database except for the migration table. This method should only be
// used during end-to-end testing.
func (s *SqlStore) Flush(ctx context.Context) {
tables, err := s.tableNames()
if err != nil {
s.log.Fatal("unable to flush sqlite", zap.Error(err))
}
for _, t := range tables {
if t == migrationsTableName {
continue
}
stmt := fmt.Sprintf("DELETE FROM %s", t)
err := s.execTrans(ctx, stmt)
if err != nil {
s.log.Fatal("unable to flush sqlite", zap.Error(err))
}
}
s.log.Debug("sqlite data flushed successfully")
}
// BackupSqlStore creates a new temporary database and uses the sqlite backup API
// to back the database up into the temporary database. It then writes the temporary
// database file to the writer. Using the sqlite backup API allows the backup to be
// performed without needing to lock the database, and also allows it to work with
// in-memory databases. See: https://www.sqlite.org/backup.html
//
// The backup works by copying the SOURCE database to the DESTINATION database.
// The SOURCE is the running database that needs to be backed up, and the DESTINATION
// is the resulting backup. The underlying sqlite connection is needed for both
// SOURCE and DESTINATION databases to use the sqlite backup API made available by the
// go-sqlite3 driver.
func (s *SqlStore) BackupSqlStore(ctx context.Context, w io.Writer) (rErr error) {
span, _ := tracing.StartSpanFromContext(ctx)
defer span.Finish()
// create a destination db in a temporary directory to hold the backup.
tempDir, err := os.MkdirTemp("", "")
if err != nil {
return err
}
defer os.RemoveAll(tempDir)
destPath := filepath.Join(tempDir, DefaultFilename)
dest, err := NewSqlStore(destPath, zap.NewNop())
if err != nil {
return err
}
defer errors2.Capture(&rErr, dest.Close)
if err := backup(ctx, dest, s); err != nil {
return err
}
// open the backup file so it can be copied to the destination writer
f, err := os.Open(destPath)
if err != nil {
return err
}
defer f.Close()
// copy the backup to the destination writer
_, err = io.Copy(w, f)
return err
}
func backup(ctx context.Context, dest, src *SqlStore) error {
// get the connection for the destination so we can get the underlying sqlite connection
destConn, err := dest.DB.Conn(ctx)
if err != nil {
return err
}
defer destConn.Close()
// get the sqlite connection for the destination to access the sqlite backup API
destSqliteConn, err := sqliteFromSqlConn(destConn)
if err != nil {
return err
}
// get the connection for the source database so we can get the underlying sqlite connection
srcConn, err := src.DB.Conn(ctx)
if err != nil {
return err
}
defer srcConn.Close()
// get the sqlite connection for the source to access the sqlite backup API
srcSqliteConn, err := sqliteFromSqlConn(srcConn)
if err != nil {
return err
}
// call Backup on the destination sqlite connection - which initializes the backup
bk, err := destSqliteConn.Backup("main", srcSqliteConn, "main")
if err != nil {
return err
}
// perform the backup
_, err = bk.Step(-1)
if err != nil {
return err
}
// close the backup once it's done
return bk.Finish()
}
// sqliteFromSqlConn returns the underlying sqlite3 connection from an sql connection
func sqliteFromSqlConn(c *sql.Conn) (*sqlite3.SQLiteConn, error) {
var sqliteConn *sqlite3.SQLiteConn
err := c.Raw(func(driverConn interface{}) error {
sqliteConn = driverConn.(*sqlite3.SQLiteConn)
return nil
})
if err != nil {
return nil, err
}
return sqliteConn, nil
}
// RestoreSqlStore replaces the underlying database with the data from r.
func (s *SqlStore) RestoreSqlStore(ctx context.Context, r io.Reader) (rErr error) {
tempDir, err := os.MkdirTemp("", "")
if err != nil {
return err
}
defer os.RemoveAll(tempDir)
tempFileName := filepath.Join(tempDir, DefaultFilename)
f, err := os.Create(tempFileName)
if err != nil {
return err
}
defer errors2.Capture(&rErr, f.Close)
// Copy the contents of r to the temporary file
if _, err := io.Copy(f, r); err != nil {
return err
}
if err := f.Sync(); err != nil {
return err
}
if err := f.Close(); err != nil {
return err
}
// Run the migrations on the restored database prior to swapping it in.
if err := s.migrateRestored(ctx, tempFileName); err != nil {
return err
}
// Use a lock while swapping over to the temporary database.
s.Mu.Lock()
defer s.Mu.Unlock()
// Close the current DB.
if err := s.Close(); err != nil {
return err
}
// If we're using a :memory: database, we need to open a new DB (which will be completely empty),
// and then use the sqlite backup API to copy the data from the restored db file into the database.
// Otherwise, we can just atomically swap the file and re-open the DB.
if s.path == InmemPath {
if err := s.openDB(); err != nil {
return err
}
// Open the temporary file - this is the "source" DB for doing the restore
tempDB, err := NewSqlStore(tempFileName, s.log.With(zap.String("service", "temp backup sqlite")))
if err != nil {
return err
}
defer tempDB.Close()
// Copy the data from the temporary restored DB into the currently open DB
return backup(ctx, s, tempDB)
}
// Atomically swap the temporary file with the current DB file.
if err := fs.RenameFileWithReplacement(tempFileName, s.path); err != nil {
return err
}
// Reopen the new database file
return s.openDB()
}
// migrateRestored opens the database at the temporary path and applies the
// migrations to it. The database at the temporary path is closed after the
// migrations are complete. This should be used as part of the restore
// operation, prior to swapping the restored database (or its contents) with the
// active database.
func (s *SqlStore) migrateRestored(ctx context.Context, tempFileName string) error {
restoredSql, err := NewSqlStore(tempFileName, s.log.With(zap.String("service", "restored sqlite")))
if err != nil {
return err
}
defer restoredSql.Close()
restoreMigrator := NewMigrator(
restoredSql,
s.log.With(zap.String("service", "sqlite restore migrations")),
)
return restoreMigrator.Up(ctx, sqliteMigrations.AllUp)
}
func (s *SqlStore) execTrans(ctx context.Context, stmt string) error {
// use a lock to prevent two potential simultaneous write operations to the database,
// which would throw an error
s.Mu.Lock()
defer s.Mu.Unlock()
tx, err := s.DB.BeginTx(ctx, nil)
if err != nil {
return err
}
_, err = tx.ExecContext(ctx, stmt)
if err != nil {
tx.Rollback()
return err
}
err = tx.Commit()
if err != nil {
return err
}
return nil
}
func (s *SqlStore) allMigrationNames() ([]string, error) {
checkStmt := fmt.Sprintf(`SELECT name FROM sqlite_master WHERE type='table' AND name='%s'`, migrationsTableName)
tbls, err := s.queryToStrings(checkStmt)
if err != nil {
return nil, err
}
if len(tbls) == 0 {
return nil, nil
}
migrStmt := fmt.Sprintf(`SELECT name FROM %s ORDER BY name`, migrationsTableName)
migr, err := s.queryToStrings(migrStmt)
if err != nil {
return nil, err
}
if len(migr) == 0 {
return nil, nil
}
return migr, nil
}
func (s *SqlStore) tableNames() ([]string, error) {
stmt := `SELECT name FROM sqlite_master WHERE type='table'`
return s.queryToStrings(stmt)
}
// helper function for running a read-only query resulting in a slice of strings from
// an arbitrary statement.
func (s *SqlStore) queryToStrings(stmt string) ([]string, error) {
var output []string
rows, err := s.DB.Query(stmt)
if err != nil {
return nil, err
}
for rows.Next() {
var i string
err = rows.Scan(&i)
if err != nil {
return nil, err
}
output = append(output, i)
}
return output, nil
}