138 lines
3.0 KiB
Go
138 lines
3.0 KiB
Go
|
// Go MySQL Driver - A MySQL-Driver for Go's database/sql package
|
||
|
//
|
||
|
// Copyright 2018 The Go-MySQL-Driver Authors. All rights reserved.
|
||
|
//
|
||
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
||
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
||
|
// You can obtain one at http://mozilla.org/MPL/2.0/.
|
||
|
|
||
|
// +build go1.10
|
||
|
|
||
|
package mysql
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"database/sql"
|
||
|
"database/sql/driver"
|
||
|
"fmt"
|
||
|
"net"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
var _ driver.DriverContext = &MySQLDriver{}
|
||
|
|
||
|
type dialCtxKey struct{}
|
||
|
|
||
|
func TestConnectorObeysDialTimeouts(t *testing.T) {
|
||
|
if !available {
|
||
|
t.Skipf("MySQL server not running on %s", netAddr)
|
||
|
}
|
||
|
|
||
|
RegisterDialContext("dialctxtest", func(ctx context.Context, addr string) (net.Conn, error) {
|
||
|
var d net.Dialer
|
||
|
if !ctx.Value(dialCtxKey{}).(bool) {
|
||
|
return nil, fmt.Errorf("test error: query context is not propagated to our dialer")
|
||
|
}
|
||
|
return d.DialContext(ctx, prot, addr)
|
||
|
})
|
||
|
|
||
|
db, err := sql.Open("mysql", fmt.Sprintf("%s:%s@dialctxtest(%s)/%s?timeout=30s", user, pass, addr, dbname))
|
||
|
if err != nil {
|
||
|
t.Fatalf("error connecting: %s", err.Error())
|
||
|
}
|
||
|
defer db.Close()
|
||
|
|
||
|
ctx := context.WithValue(context.Background(), dialCtxKey{}, true)
|
||
|
|
||
|
_, err = db.ExecContext(ctx, "DO 1")
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func configForTests(t *testing.T) *Config {
|
||
|
if !available {
|
||
|
t.Skipf("MySQL server not running on %s", netAddr)
|
||
|
}
|
||
|
|
||
|
mycnf := NewConfig()
|
||
|
mycnf.User = user
|
||
|
mycnf.Passwd = pass
|
||
|
mycnf.Addr = addr
|
||
|
mycnf.Net = prot
|
||
|
mycnf.DBName = dbname
|
||
|
return mycnf
|
||
|
}
|
||
|
|
||
|
func TestNewConnector(t *testing.T) {
|
||
|
mycnf := configForTests(t)
|
||
|
conn, err := NewConnector(mycnf)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
|
||
|
db := sql.OpenDB(conn)
|
||
|
defer db.Close()
|
||
|
|
||
|
if err := db.Ping(); err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
type slowConnection struct {
|
||
|
net.Conn
|
||
|
slowdown time.Duration
|
||
|
}
|
||
|
|
||
|
func (sc *slowConnection) Read(b []byte) (int, error) {
|
||
|
time.Sleep(sc.slowdown)
|
||
|
return sc.Conn.Read(b)
|
||
|
}
|
||
|
|
||
|
type connectorHijack struct {
|
||
|
driver.Connector
|
||
|
connErr error
|
||
|
}
|
||
|
|
||
|
func (cw *connectorHijack) Connect(ctx context.Context) (driver.Conn, error) {
|
||
|
var conn driver.Conn
|
||
|
conn, cw.connErr = cw.Connector.Connect(ctx)
|
||
|
return conn, cw.connErr
|
||
|
}
|
||
|
|
||
|
func TestConnectorTimeoutsDuringOpen(t *testing.T) {
|
||
|
RegisterDialContext("slowconn", func(ctx context.Context, addr string) (net.Conn, error) {
|
||
|
var d net.Dialer
|
||
|
conn, err := d.DialContext(ctx, prot, addr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return &slowConnection{Conn: conn, slowdown: 100 * time.Millisecond}, nil
|
||
|
})
|
||
|
|
||
|
mycnf := configForTests(t)
|
||
|
mycnf.Net = "slowconn"
|
||
|
|
||
|
conn, err := NewConnector(mycnf)
|
||
|
if err != nil {
|
||
|
t.Fatal(err)
|
||
|
}
|
||
|
|
||
|
hijack := &connectorHijack{Connector: conn}
|
||
|
|
||
|
db := sql.OpenDB(hijack)
|
||
|
defer db.Close()
|
||
|
|
||
|
ctx, cancel := context.WithTimeout(context.Background(), 50*time.Millisecond)
|
||
|
defer cancel()
|
||
|
|
||
|
_, err = db.ExecContext(ctx, "DO 1")
|
||
|
if err != context.DeadlineExceeded {
|
||
|
t.Fatalf("ExecContext should have timed out")
|
||
|
}
|
||
|
if hijack.connErr != context.DeadlineExceeded {
|
||
|
t.Fatalf("(*Connector).Connect should have timed out")
|
||
|
}
|
||
|
}
|