-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtx.go
97 lines (86 loc) · 3.39 KB
/
tx.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
package dali
import (
"context"
"database/sql"
"github.com/mibk/dali/dialect"
)
// Tx wraps the sql.Tx to provide the Query method instead
// of the sql.Tx's original methods for comunication with
// the database.
type Tx struct {
Tx *sql.Tx
dialect dialect.Dialect
middleware func(Execer) Execer
}
// QueryWithContext is a (*DB).Query equivalent for transactions.
func (tx *Tx) QueryWithContext(ctx context.Context, query string, args ...interface{}) *Query {
sql, err := translate(tx.dialect, query, args)
return &Query{
ctx: ctx,
execer: tx.middleware(tx.Tx),
query: sql,
err: err,
}
}
// Query is a (*DB).Query equivalent for transactions.
func (tx *Tx) Query(query string, args ...interface{}) *Query {
return tx.QueryWithContext(context.Background(), query, args...)
}
// PrepareContext creates a prepared statement for later queries or executions.
// The caller must call the statement's Close method when the statement
// is no longer needed. Unlike the Prepare methods in database/sql this
// method also accepts args, which are meant only for query building.
// Therefore, only ?ident, ?ident..., ?sql are interpolated in this phase.
// Apart of that, ? is the only other placeholder allowed (this one
// will be transformed into a dialect specific one to allow the parameter
// binding.
//
// The provided context is used for the preparation of the statement, not
// for the execution of the statement.
func (tx *Tx) PrepareContext(ctx context.Context, query string, args ...interface{}) (*Stmt, error) {
sql, err := translatePreparedStmt(tx.dialect, query, args)
if err != nil {
return nil, err
}
stmt, err := tx.Tx.PrepareContext(ctx, sql)
if err != nil {
return nil, err
}
return &Stmt{stmt, sql, tx.middleware}, nil
}
// Prepare creates a prepared statement for later queries or executions.
// The caller must call the statement's Close method when the statement
// is no longer needed. Unlike the Prepare methods in database/sql this
// method also accepts args, which are meant only for query building.
// Therefore, only ?ident, ?ident..., ?sql are interpolated in this phase.
// Apart of that, ? is the only other placeholder allowed (this one
// will be transformed into a dialect specific one to allow the parameter
// binding.
func (tx *Tx) Prepare(query string, args ...interface{}) (*Stmt, error) {
return tx.PrepareContext(context.Background(), query, args...)
}
// Stmt returns a transaction-specific prepared statement from an
// existing statement.
//
// The provided context is used for the preparation of the statement,
// not for the execution of the statement.
//
// The returned statement operates within the transaction and can no
// longer be used once the transaction has been committed or rolled
// back.
func (tx *Tx) StmtContext(ctx context.Context, stmt *Stmt) *Stmt {
stmt2 := new(Stmt)
*stmt2 = *stmt
stmt2.stmt = tx.Tx.StmtContext(ctx, stmt2.stmt)
return stmt2
}
// Stmt returns a transaction-specific prepared statement from an existing
// statement. The returned statement operates within the transaction and can
// no longer be used once the transaction has been committed or rolled back.
func (tx *Tx) Stmt(stmt *Stmt) *Stmt {
return tx.StmtContext(context.Background(), stmt)
}
// Commit commits the transaction.
func (tx *Tx) Commit() error { return tx.Tx.Commit() }
// Rollback aborts the transaction.
func (tx *Tx) Rollback() error { return tx.Tx.Rollback() }