123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- // Copyright (c) 2012-present The upper.io/db authors. All rights reserved.
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
-
- package sqladapter
-
- import (
- "context"
- "database/sql"
- "sync/atomic"
-
- "upper.io/db.v3"
- "upper.io/db.v3/lib/sqlbuilder"
- )
-
- // DatabaseTx represents a database session within a transaction.
- type DatabaseTx interface {
- BaseDatabase
- PartialDatabase
-
- BaseTx
- }
-
- // BaseTx provides logic for methods that can be shared across all SQL
- // adapters.
- type BaseTx interface {
- db.Tx
-
- // Committed returns true if the transaction was already commited.
- Committed() bool
- }
-
- type databaseTx struct {
- Database
- BaseTx
- }
-
- // NewDatabaseTx creates a database session within a transaction.
- func NewDatabaseTx(db Database) DatabaseTx {
- return &databaseTx{
- Database: db,
- BaseTx: db.Transaction(),
- }
- }
-
- type baseTx struct {
- *sql.Tx
- committed atomic.Value
- }
-
- func newBaseTx(tx *sql.Tx) BaseTx {
- return &baseTx{Tx: tx}
- }
-
- func (b *baseTx) Committed() bool {
- committed := b.committed.Load()
- if committed != nil {
- return true
- }
- return false
- }
-
- func (b *baseTx) Commit() (err error) {
- err = b.Tx.Commit()
- if err != nil {
- return err
- }
- b.committed.Store(struct{}{})
- return nil
- }
-
- func (w *databaseTx) Commit() error {
- defer w.Database.Close() // Automatic close on commit.
- return w.BaseTx.Commit()
- }
-
- func (w *databaseTx) Rollback() error {
- defer w.Database.Close() // Automatic close on rollback.
- return w.BaseTx.Rollback()
- }
-
- // RunTx creates a transaction context and runs fn within it.
- func RunTx(d sqlbuilder.Database, ctx context.Context, fn func(tx sqlbuilder.Tx) error) error {
- tx, err := d.NewTx(ctx)
- if err != nil {
- return err
- }
-
- defer tx.Close()
- if err := fn(tx); err != nil {
- tx.Rollback()
- return err
- }
- return tx.Commit()
- }
-
- var (
- _ = BaseTx(&baseTx{})
- _ = DatabaseTx(&databaseTx{})
- )
|