2017-02-12 13:13:54 +02:00
|
|
|
// This package implements database/sql/driver interface,
|
|
|
|
// so we can use go-mysql with database/sql
|
|
|
|
package driver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
sqldriver "database/sql/driver"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/juju/errors"
|
|
|
|
"github.com/siddontang/go-mysql/client"
|
|
|
|
"github.com/siddontang/go-mysql/mysql"
|
|
|
|
"github.com/siddontang/go/hack"
|
|
|
|
)
|
|
|
|
|
|
|
|
type driver struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
// DSN user:password@addr[?db]
|
|
|
|
func (d driver) Open(dsn string) (sqldriver.Conn, error) {
|
2019-01-01 10:57:46 +02:00
|
|
|
lastIndex := strings.LastIndex(dsn, "@")
|
|
|
|
seps := []string{dsn[:lastIndex], dsn[lastIndex+1:]}
|
2017-02-12 13:13:54 +02:00
|
|
|
if len(seps) != 2 {
|
|
|
|
return nil, errors.Errorf("invalid dsn, must user:password@addr[?db]")
|
|
|
|
}
|
|
|
|
|
|
|
|
var user string
|
|
|
|
var password string
|
|
|
|
var addr string
|
|
|
|
var db string
|
|
|
|
|
|
|
|
if ss := strings.Split(seps[0], ":"); len(ss) == 2 {
|
|
|
|
user, password = ss[0], ss[1]
|
|
|
|
} else if len(ss) == 1 {
|
|
|
|
user = ss[0]
|
|
|
|
} else {
|
|
|
|
return nil, errors.Errorf("invalid dsn, must user:password@addr[?db]")
|
|
|
|
}
|
|
|
|
|
|
|
|
if ss := strings.Split(seps[1], "?"); len(ss) == 2 {
|
|
|
|
addr, db = ss[0], ss[1]
|
|
|
|
} else if len(ss) == 1 {
|
|
|
|
addr = ss[0]
|
|
|
|
} else {
|
|
|
|
return nil, errors.Errorf("invalid dsn, must user:password@addr[?db]")
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := client.Connect(addr, user, password, db)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &conn{c}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type conn struct {
|
|
|
|
*client.Conn
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *conn) Prepare(query string) (sqldriver.Stmt, error) {
|
|
|
|
st, err := c.Conn.Prepare(query)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Trace(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &stmt{st}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *conn) Close() error {
|
|
|
|
return c.Conn.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *conn) Begin() (sqldriver.Tx, error) {
|
|
|
|
err := c.Conn.Begin()
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Trace(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &tx{c.Conn}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildArgs(args []sqldriver.Value) []interface{} {
|
|
|
|
a := make([]interface{}, len(args))
|
|
|
|
|
|
|
|
for i, arg := range args {
|
|
|
|
a[i] = arg
|
|
|
|
}
|
|
|
|
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
|
|
|
|
func replyError(err error) error {
|
|
|
|
if mysql.ErrorEqual(err, mysql.ErrBadConn) {
|
|
|
|
return sqldriver.ErrBadConn
|
|
|
|
} else {
|
|
|
|
return errors.Trace(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *conn) Exec(query string, args []sqldriver.Value) (sqldriver.Result, error) {
|
|
|
|
a := buildArgs(args)
|
|
|
|
r, err := c.Conn.Execute(query, a...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, replyError(err)
|
|
|
|
}
|
|
|
|
return &result{r}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *conn) Query(query string, args []sqldriver.Value) (sqldriver.Rows, error) {
|
|
|
|
a := buildArgs(args)
|
|
|
|
r, err := c.Conn.Execute(query, a...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, replyError(err)
|
|
|
|
}
|
|
|
|
return newRows(r.Resultset)
|
|
|
|
}
|
|
|
|
|
|
|
|
type stmt struct {
|
|
|
|
*client.Stmt
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stmt) Close() error {
|
|
|
|
return s.Stmt.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stmt) NumInput() int {
|
|
|
|
return s.Stmt.ParamNum()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stmt) Exec(args []sqldriver.Value) (sqldriver.Result, error) {
|
|
|
|
a := buildArgs(args)
|
|
|
|
r, err := s.Stmt.Execute(a...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, replyError(err)
|
|
|
|
}
|
|
|
|
return &result{r}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *stmt) Query(args []sqldriver.Value) (sqldriver.Rows, error) {
|
|
|
|
a := buildArgs(args)
|
|
|
|
r, err := s.Stmt.Execute(a...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, replyError(err)
|
|
|
|
}
|
|
|
|
return newRows(r.Resultset)
|
|
|
|
}
|
|
|
|
|
|
|
|
type tx struct {
|
|
|
|
*client.Conn
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *tx) Commit() error {
|
|
|
|
return t.Conn.Commit()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *tx) Rollback() error {
|
|
|
|
return t.Conn.Rollback()
|
|
|
|
}
|
|
|
|
|
|
|
|
type result struct {
|
|
|
|
*mysql.Result
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *result) LastInsertId() (int64, error) {
|
|
|
|
return int64(r.Result.InsertId), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *result) RowsAffected() (int64, error) {
|
|
|
|
return int64(r.Result.AffectedRows), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type rows struct {
|
|
|
|
*mysql.Resultset
|
|
|
|
|
|
|
|
columns []string
|
|
|
|
step int
|
|
|
|
}
|
|
|
|
|
|
|
|
func newRows(r *mysql.Resultset) (*rows, error) {
|
|
|
|
if r == nil {
|
|
|
|
return nil, fmt.Errorf("invalid mysql query, no correct result")
|
|
|
|
}
|
|
|
|
|
|
|
|
rs := new(rows)
|
|
|
|
rs.Resultset = r
|
|
|
|
|
|
|
|
rs.columns = make([]string, len(r.Fields))
|
|
|
|
|
|
|
|
for i, f := range r.Fields {
|
|
|
|
rs.columns[i] = hack.String(f.Name)
|
|
|
|
}
|
|
|
|
rs.step = 0
|
|
|
|
|
|
|
|
return rs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Columns() []string {
|
|
|
|
return r.columns
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Close() error {
|
|
|
|
r.step = -1
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *rows) Next(dest []sqldriver.Value) error {
|
|
|
|
if r.step >= r.Resultset.RowNumber() {
|
|
|
|
return io.EOF
|
|
|
|
} else if r.step == -1 {
|
|
|
|
return io.ErrUnexpectedEOF
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < r.Resultset.ColumnNumber(); i++ {
|
|
|
|
value, err := r.Resultset.GetValue(r.step, i)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
dest[i] = sqldriver.Value(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
r.step++
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
sql.Register("mysql", driver{})
|
|
|
|
}
|