2018-06-23 21:46:35 +00:00
|
|
|
package sqldb
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"database/sql"
|
|
|
|
|
|
|
|
"github.com/juju/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Conn represents database connection.
|
2018-06-24 22:22:27 +00:00
|
|
|
type Conn interface {
|
|
|
|
connOrTx
|
|
|
|
Begin(context.Context, func(Tx) error) error
|
|
|
|
Close() error
|
|
|
|
DB() *sql.DB
|
|
|
|
Before(BeforeCallback)
|
|
|
|
After(AfterCallback)
|
|
|
|
}
|
2018-06-23 21:46:35 +00:00
|
|
|
|
2018-06-24 22:22:27 +00:00
|
|
|
// Tx represents database transacation.
|
|
|
|
type Tx interface {
|
|
|
|
connOrTx
|
|
|
|
Commit() error
|
|
|
|
Rollback() error
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
2018-06-24 22:22:27 +00:00
|
|
|
type dbWrapper struct {
|
|
|
|
conn *sql.DB
|
|
|
|
*caller
|
|
|
|
}
|
2018-06-23 21:46:35 +00:00
|
|
|
|
|
|
|
// Flavor defines a kind of SQL database.
|
|
|
|
type Flavor string
|
|
|
|
|
|
|
|
const (
|
|
|
|
// MySQL is the MySQL SQL flavor.
|
|
|
|
MySQL Flavor = "mysql"
|
|
|
|
// PostgreSQL is the PostgreSQL SQL flavor.
|
|
|
|
PostgreSQL Flavor = "postgresql"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Connect establishes a new database connection.
|
2018-06-24 22:22:27 +00:00
|
|
|
func Connect(ctx context.Context, f Flavor, dsn string) (Conn, error) {
|
2018-06-23 21:46:35 +00:00
|
|
|
conn, err := sql.Open(string(f), dsn)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Annotate(err, "Failed to establish connection")
|
|
|
|
}
|
|
|
|
err = conn.PingContext(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Annotate(err, "Connection is not responding")
|
|
|
|
}
|
2018-06-24 22:22:27 +00:00
|
|
|
return &dbWrapper{
|
|
|
|
conn: conn,
|
|
|
|
caller: &caller{
|
|
|
|
db: conn,
|
|
|
|
cb: &callbacks{},
|
|
|
|
},
|
|
|
|
}, nil
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
2018-06-24 22:22:27 +00:00
|
|
|
// Begin executes a transaction.
|
|
|
|
func (c *dbWrapper) Begin(ctx context.Context, fn func(tx Tx) error) error {
|
|
|
|
tx, err := c.conn.BeginTx(ctx, &sql.TxOptions{})
|
2018-06-23 21:46:35 +00:00
|
|
|
if err != nil {
|
2018-06-24 22:22:27 +00:00
|
|
|
return err
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
2018-06-24 22:22:27 +00:00
|
|
|
err = fn(c.wrapTx(tx))
|
|
|
|
if err != nil {
|
|
|
|
tx.Rollback()
|
|
|
|
}
|
|
|
|
return err
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DB returns the underlying DB object.
|
2018-06-24 22:22:27 +00:00
|
|
|
func (c *dbWrapper) DB() *sql.DB {
|
|
|
|
return c.conn
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the connection.
|
2018-06-24 22:22:27 +00:00
|
|
|
func (c *dbWrapper) Close() error {
|
|
|
|
return c.conn.Close()
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Before adds a callback function that would be called before a query is
|
|
|
|
// executed.
|
2018-06-24 22:22:27 +00:00
|
|
|
func (c *dbWrapper) Before(cb BeforeCallback) {
|
|
|
|
c.cb.addBefore(cb)
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// After adds a callback function that would be called after a query was
|
|
|
|
// executed.
|
2018-06-24 22:22:27 +00:00
|
|
|
func (c *dbWrapper) After(cb AfterCallback) {
|
|
|
|
c.cb.addAfter(cb)
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
|
2018-06-24 22:22:27 +00:00
|
|
|
func (c *dbWrapper) wrapTx(tx *sql.Tx) Tx {
|
|
|
|
return &txWrapper{
|
|
|
|
tx: tx,
|
|
|
|
connOrTx: &caller{
|
|
|
|
db: tx,
|
|
|
|
cb: c.cb,
|
|
|
|
},
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-24 22:22:27 +00:00
|
|
|
type txWrapper struct {
|
|
|
|
tx *sql.Tx
|
|
|
|
connOrTx
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *txWrapper) Commit() error {
|
|
|
|
return w.tx.Commit()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *txWrapper) Rollback() error {
|
|
|
|
return w.tx.Rollback()
|
2018-06-23 21:46:35 +00:00
|
|
|
}
|