bilibili-backup/library/database/tidb/tidb_test.go

235 lines
5.7 KiB
Go
Raw Permalink Normal View History

2019-04-22 10:59:20 +08:00
package tidb
import (
"context"
"database/sql"
"os"
"testing"
"time"
"go-common/library/net/netutil/breaker"
xtime "go-common/library/time"
)
func TestMySQL(t *testing.T) {
bc := &breaker.Config{
Window: xtime.Duration(10 * time.Second),
Sleep: xtime.Duration(10 * time.Second),
Bucket: 10,
Ratio: 0.5,
Request: 100,
}
dsn := os.Getenv("TEST_MYSQL_DSN")
if dsn == "" {
t.Skipf("TEST_MYSQL_DSN is empty, sql test skipped")
}
dsn = dsn + "?timeout=5s&readTimeout=5s&writeTimeout=5s&parseTime=true&loc=Local&charset=utf8"
c := &Config{
DSN: dsn,
Active: 10,
Idle: 5,
IdleTimeout: xtime.Duration(time.Minute),
QueryTimeout: xtime.Duration(time.Minute),
ExecTimeout: xtime.Duration(time.Minute),
TranTimeout: xtime.Duration(time.Minute),
Breaker: bc,
}
db := NewTiDB(c)
defer db.Close()
testPing(t, db)
testTable(t, db)
testExec(t, db)
testQuery(t, db)
testQueryRow(t, db)
testPrepare(t, db)
testPrepared(t, db)
testTransaction(t, db)
}
func testTransaction(t *testing.T, db *DB) {
var (
tx *Tx
err error
execSQL = "INSERT INTO test(name) VALUES(?)"
selSQL = "SELECT name FROM test WHERE name=?"
txstmt *Stmt
)
if tx, err = db.Begin(context.TODO()); err != nil {
t.Errorf("MySQL: db transaction Begin err(%v)", err)
tx.Rollback()
return
}
t.Log("MySQL: db transaction begin")
if txstmt, err = tx.Prepare(execSQL); err != nil {
t.Errorf("MySQL: tx.Prepare err(%v)", err)
}
if stmt := tx.Stmt(txstmt); stmt == nil {
t.Errorf("MySQL:tx.Stmt err(%v)", err)
}
// exec
if _, err = tx.Exec(execSQL, "tx1"); err != nil {
t.Errorf("MySQL: tx.Exec err(%v)", err)
tx.Rollback()
return
}
t.Logf("MySQL:tx.Exec tx1")
if _, err = tx.Exec(execSQL, "tx1"); err != nil {
t.Errorf("MySQL: tx.Exec err(%v)", err)
tx.Rollback()
return
}
t.Logf("MySQL:tx.Exec tx1")
// query
rows, err := tx.Query(selSQL, "tx2")
if err != nil {
t.Errorf("MySQL:tx.Query err(%v)", err)
tx.Rollback()
return
}
rows.Close()
t.Log("MySQL: tx.Query tx2")
// queryrow
var name string
row := tx.QueryRow(selSQL, "noexist")
if err = row.Scan(&name); err != sql.ErrNoRows {
t.Errorf("MySQL: queryRow name: noexist")
}
if err = tx.Commit(); err != nil {
t.Errorf("MySQL:tx.Commit err(%v)", err)
}
if err = tx.Commit(); err != nil {
t.Logf("MySQL:tx.Commit err(%v)", err)
}
if err = tx.Rollback(); err != nil {
t.Logf("MySQL:tx Rollback err(%v)", err)
}
}
func testPing(t *testing.T, db *DB) {
if err := db.Ping(context.TODO()); err != nil {
t.Errorf("MySQL: ping error(%v)", err)
t.FailNow()
} else {
t.Log("MySQL: ping ok")
}
}
func testTable(t *testing.T, db *DB) {
table := "CREATE TABLE IF NOT EXISTS `test` (`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增ID', `name` varchar(16) NOT NULL DEFAULT '' COMMENT '名称', PRIMARY KEY (`id`)) ENGINE=InnoDB DEFAULT CHARSET=utf8"
if _, err := db.Exec(context.TODO(), table); err != nil {
t.Errorf("MySQL: create table error(%v)", err)
} else {
t.Log("MySQL: create table ok")
}
}
func testExec(t *testing.T, db *DB) {
sql := "INSERT INTO test(name) VALUES(?)"
if _, err := db.Exec(context.TODO(), sql, "test"); err != nil {
t.Errorf("MySQL: insert error(%v)", err)
} else {
t.Log("MySQL: insert ok")
}
}
func testQuery(t *testing.T, db *DB) {
sql := "SELECT name FROM test WHERE name=?"
rows, err := db.Query(context.TODO(), sql, "test")
if err != nil {
t.Errorf("MySQL: query error(%v)", err)
}
defer rows.Close()
for rows.Next() {
name := ""
if err := rows.Scan(&name); err != nil {
t.Errorf("MySQL: query scan error(%v)", err)
} else {
t.Logf("MySQL: query name: %s", name)
}
}
}
func testQueryRow(t *testing.T, db *DB) {
sql := "SELECT name FROM test WHERE name=?"
name := ""
row := db.QueryRow(context.TODO(), sql, "test")
if err := row.Scan(&name); err != nil {
t.Errorf("MySQL: queryRow error(%v)", err)
} else {
t.Logf("MySQL: queryRow name: %s", name)
}
}
func testPrepared(t *testing.T, db *DB) {
sql := "SELECT name FROM test WHERE name=?"
name := ""
stmt := db.Prepared(sql)
row := stmt.QueryRow(context.TODO(), "test")
if err := row.Scan(&name); err != nil {
t.Errorf("MySQL: prepared query error(%v)", err)
} else {
t.Logf("MySQL: prepared query name: %s", name)
}
if err := stmt.Close(); err != nil {
t.Errorf("MySQL:stmt.Close err(%v)", err)
}
}
func testPrepare(t *testing.T, db *DB) {
var (
selsql = "SELECT name FROM test WHERE name=?"
execsql = "INSERT INTO test(name) VALUES(?)"
name = ""
)
selstmt, err := db.Prepare(selsql)
if err != nil {
t.Errorf("MySQL:Prepare err(%v)", err)
return
}
row := selstmt.QueryRow(context.TODO(), "noexit")
if err = row.Scan(&name); err == sql.ErrNoRows {
t.Logf("MySQL: prepare query error(%v)", err)
} else {
t.Errorf("MySQL: prepared query name: noexist")
}
rows, err := selstmt.Query(context.TODO(), "test")
if err != nil {
t.Errorf("MySQL:stmt.Query err(%v)", err)
}
rows.Close()
execstmt, err := db.Prepare(execsql)
if err != nil {
t.Errorf("MySQL:Prepare err(%v)", err)
return
}
if _, err := execstmt.Exec(context.TODO(), "test"); err != nil {
t.Errorf("MySQL: stmt.Exec(%v)", err)
}
}
func BenchmarkMySQL(b *testing.B) {
c := &Config{
DSN: "test:test@tcp(172.16.0.148:3306)/test?timeout=5s&readTimeout=5s&writeTimeout=5s&parseTime=true&loc=Local&charset=utf8",
Active: 10,
Idle: 5,
IdleTimeout: xtime.Duration(time.Minute),
}
db := NewTiDB(c)
defer db.Close()
b.RunParallel(func(pb *testing.PB) {
for pb.Next() {
sql := "SELECT name FROM test WHERE name=?"
rows, err := db.Query(context.TODO(), sql, "test")
if err == nil {
for rows.Next() {
var name string
if err = rows.Scan(&name); err != nil {
break
}
}
rows.Close()
}
}
})
}