2022-04-06 10:36:06 -04:00

123 lines
2.8 KiB
Go

package sqlds
import (
"context"
"database/sql"
"fmt"
datastore "github.com/ipfs/go-datastore"
dsq "github.com/ipfs/go-datastore/query"
)
// ErrNotImplemented is returned when the SQL datastore does not yet implement the function call.
var ErrNotImplemented = fmt.Errorf("not implemented")
type txn struct {
db *sql.DB
queries Queries
txn *sql.Tx
}
// NewTransaction creates a new database transaction, note the readOnly parameter is ignored by this implementation.
func (ds *Datastore) NewTransaction(ctx context.Context, _ bool) (datastore.Txn, error) {
sqlTxn, err := ds.db.BeginTx(ctx, nil)
if err != nil {
if sqlTxn != nil {
// nothing we can do about this error.
_ = sqlTxn.Rollback()
}
return nil, err
}
return &txn{
db: ds.db,
queries: ds.queries,
txn: sqlTxn,
}, nil
}
func (t *txn) Get(ctx context.Context, key datastore.Key) ([]byte, error) {
row := t.txn.QueryRowContext(ctx, t.queries.Get(), key.String())
var out []byte
switch err := row.Scan(&out); err {
case sql.ErrNoRows:
return nil, datastore.ErrNotFound
case nil:
return out, nil
default:
return nil, err
}
}
func (t *txn) Has(ctx context.Context, key datastore.Key) (bool, error) {
row := t.txn.QueryRowContext(ctx, t.queries.Exists(), key.String())
var exists bool
switch err := row.Scan(&exists); err {
case sql.ErrNoRows:
return exists, nil
case nil:
return exists, nil
default:
return exists, err
}
}
func (t *txn) GetSize(ctx context.Context, key datastore.Key) (int, error) {
row := t.txn.QueryRowContext(ctx, t.queries.GetSize(), key.String())
var size int
switch err := row.Scan(&size); err {
case sql.ErrNoRows:
return -1, datastore.ErrNotFound
case nil:
return size, nil
default:
return 0, err
}
}
func (t *txn) Query(ctx context.Context, q dsq.Query) (dsq.Results, error) {
return nil, ErrNotImplemented
}
// Put adds a value to the datastore identified by the given key.
func (t *txn) Put(ctx context.Context, key datastore.Key, val []byte) error {
_, err := t.txn.ExecContext(ctx, t.queries.Put(), key.String(), val)
if err != nil {
_ = t.txn.Rollback()
return err
}
return nil
}
// Delete removes a value from the datastore that matches the given key.
func (t *txn) Delete(ctx context.Context, key datastore.Key) error {
_, err := t.txn.ExecContext(ctx, t.queries.Delete(), key.String())
if err != nil {
_ = t.txn.Rollback()
return err
}
return nil
}
// Commit finalizes a transaction.
func (t *txn) Commit(ctx context.Context) error {
err := t.txn.Commit()
if err != nil {
_ = t.txn.Rollback()
return err
}
return nil
}
// Discard throws away changes recorded in a transaction without committing
// them to the underlying Datastore.
func (t *txn) Discard(ctx context.Context) {
_ = t.txn.Rollback()
}
var _ datastore.TxnDatastore = (*Datastore)(nil)