335 lines
8.9 KiB
Go
335 lines
8.9 KiB
Go
|
// Go MySQL Driver - A MySQL-Driver for Go's database/sql package
|
||
|
//
|
||
|
// Copyright 2013 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/.
|
||
|
|
||
|
package mysql
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"database/sql"
|
||
|
"database/sql/driver"
|
||
|
"encoding/binary"
|
||
|
"testing"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
func TestScanNullTime(t *testing.T) {
|
||
|
var scanTests = []struct {
|
||
|
in interface{}
|
||
|
error bool
|
||
|
valid bool
|
||
|
time time.Time
|
||
|
}{
|
||
|
{tDate, false, true, tDate},
|
||
|
{sDate, false, true, tDate},
|
||
|
{[]byte(sDate), false, true, tDate},
|
||
|
{tDateTime, false, true, tDateTime},
|
||
|
{sDateTime, false, true, tDateTime},
|
||
|
{[]byte(sDateTime), false, true, tDateTime},
|
||
|
{tDate0, false, true, tDate0},
|
||
|
{sDate0, false, true, tDate0},
|
||
|
{[]byte(sDate0), false, true, tDate0},
|
||
|
{sDateTime0, false, true, tDate0},
|
||
|
{[]byte(sDateTime0), false, true, tDate0},
|
||
|
{"", true, false, tDate0},
|
||
|
{"1234", true, false, tDate0},
|
||
|
{0, true, false, tDate0},
|
||
|
}
|
||
|
|
||
|
var nt = NullTime{}
|
||
|
var err error
|
||
|
|
||
|
for _, tst := range scanTests {
|
||
|
err = nt.Scan(tst.in)
|
||
|
if (err != nil) != tst.error {
|
||
|
t.Errorf("%v: expected error status %t, got %t", tst.in, tst.error, (err != nil))
|
||
|
}
|
||
|
if nt.Valid != tst.valid {
|
||
|
t.Errorf("%v: expected valid status %t, got %t", tst.in, tst.valid, nt.Valid)
|
||
|
}
|
||
|
if nt.Time != tst.time {
|
||
|
t.Errorf("%v: expected time %v, got %v", tst.in, tst.time, nt.Time)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestLengthEncodedInteger(t *testing.T) {
|
||
|
var integerTests = []struct {
|
||
|
num uint64
|
||
|
encoded []byte
|
||
|
}{
|
||
|
{0x0000000000000000, []byte{0x00}},
|
||
|
{0x0000000000000012, []byte{0x12}},
|
||
|
{0x00000000000000fa, []byte{0xfa}},
|
||
|
{0x0000000000000100, []byte{0xfc, 0x00, 0x01}},
|
||
|
{0x0000000000001234, []byte{0xfc, 0x34, 0x12}},
|
||
|
{0x000000000000ffff, []byte{0xfc, 0xff, 0xff}},
|
||
|
{0x0000000000010000, []byte{0xfd, 0x00, 0x00, 0x01}},
|
||
|
{0x0000000000123456, []byte{0xfd, 0x56, 0x34, 0x12}},
|
||
|
{0x0000000000ffffff, []byte{0xfd, 0xff, 0xff, 0xff}},
|
||
|
{0x0000000001000000, []byte{0xfe, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00}},
|
||
|
{0x123456789abcdef0, []byte{0xfe, 0xf0, 0xde, 0xbc, 0x9a, 0x78, 0x56, 0x34, 0x12}},
|
||
|
{0xffffffffffffffff, []byte{0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff}},
|
||
|
}
|
||
|
|
||
|
for _, tst := range integerTests {
|
||
|
num, isNull, numLen := readLengthEncodedInteger(tst.encoded)
|
||
|
if isNull {
|
||
|
t.Errorf("%x: expected %d, got NULL", tst.encoded, tst.num)
|
||
|
}
|
||
|
if num != tst.num {
|
||
|
t.Errorf("%x: expected %d, got %d", tst.encoded, tst.num, num)
|
||
|
}
|
||
|
if numLen != len(tst.encoded) {
|
||
|
t.Errorf("%x: expected size %d, got %d", tst.encoded, len(tst.encoded), numLen)
|
||
|
}
|
||
|
encoded := appendLengthEncodedInteger(nil, num)
|
||
|
if !bytes.Equal(encoded, tst.encoded) {
|
||
|
t.Errorf("%v: expected %x, got %x", num, tst.encoded, encoded)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestFormatBinaryDateTime(t *testing.T) {
|
||
|
rawDate := [11]byte{}
|
||
|
binary.LittleEndian.PutUint16(rawDate[:2], 1978) // years
|
||
|
rawDate[2] = 12 // months
|
||
|
rawDate[3] = 30 // days
|
||
|
rawDate[4] = 15 // hours
|
||
|
rawDate[5] = 46 // minutes
|
||
|
rawDate[6] = 23 // seconds
|
||
|
binary.LittleEndian.PutUint32(rawDate[7:], 987654) // microseconds
|
||
|
expect := func(expected string, inlen, outlen uint8) {
|
||
|
actual, _ := formatBinaryDateTime(rawDate[:inlen], outlen)
|
||
|
bytes, ok := actual.([]byte)
|
||
|
if !ok {
|
||
|
t.Errorf("formatBinaryDateTime must return []byte, was %T", actual)
|
||
|
}
|
||
|
if string(bytes) != expected {
|
||
|
t.Errorf(
|
||
|
"expected %q, got %q for length in %d, out %d",
|
||
|
expected, actual, inlen, outlen,
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
expect("0000-00-00", 0, 10)
|
||
|
expect("0000-00-00 00:00:00", 0, 19)
|
||
|
expect("1978-12-30", 4, 10)
|
||
|
expect("1978-12-30 15:46:23", 7, 19)
|
||
|
expect("1978-12-30 15:46:23.987654", 11, 26)
|
||
|
}
|
||
|
|
||
|
func TestFormatBinaryTime(t *testing.T) {
|
||
|
expect := func(expected string, src []byte, outlen uint8) {
|
||
|
actual, _ := formatBinaryTime(src, outlen)
|
||
|
bytes, ok := actual.([]byte)
|
||
|
if !ok {
|
||
|
t.Errorf("formatBinaryDateTime must return []byte, was %T", actual)
|
||
|
}
|
||
|
if string(bytes) != expected {
|
||
|
t.Errorf(
|
||
|
"expected %q, got %q for src=%q and outlen=%d",
|
||
|
expected, actual, src, outlen)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// binary format:
|
||
|
// sign (0: positive, 1: negative), days(4), hours, minutes, seconds, micro(4)
|
||
|
|
||
|
// Zeros
|
||
|
expect("00:00:00", []byte{}, 8)
|
||
|
expect("00:00:00.0", []byte{}, 10)
|
||
|
expect("00:00:00.000000", []byte{}, 15)
|
||
|
|
||
|
// Without micro(4)
|
||
|
expect("12:34:56", []byte{0, 0, 0, 0, 0, 12, 34, 56}, 8)
|
||
|
expect("-12:34:56", []byte{1, 0, 0, 0, 0, 12, 34, 56}, 8)
|
||
|
expect("12:34:56.00", []byte{0, 0, 0, 0, 0, 12, 34, 56}, 11)
|
||
|
expect("24:34:56", []byte{0, 1, 0, 0, 0, 0, 34, 56}, 8)
|
||
|
expect("-99:34:56", []byte{1, 4, 0, 0, 0, 3, 34, 56}, 8)
|
||
|
expect("103079215103:34:56", []byte{0, 255, 255, 255, 255, 23, 34, 56}, 8)
|
||
|
|
||
|
// With micro(4)
|
||
|
expect("12:34:56.00", []byte{0, 0, 0, 0, 0, 12, 34, 56, 99, 0, 0, 0}, 11)
|
||
|
expect("12:34:56.000099", []byte{0, 0, 0, 0, 0, 12, 34, 56, 99, 0, 0, 0}, 15)
|
||
|
}
|
||
|
|
||
|
func TestEscapeBackslash(t *testing.T) {
|
||
|
expect := func(expected, value string) {
|
||
|
actual := string(escapeBytesBackslash([]byte{}, []byte(value)))
|
||
|
if actual != expected {
|
||
|
t.Errorf(
|
||
|
"expected %s, got %s",
|
||
|
expected, actual,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
actual = string(escapeStringBackslash([]byte{}, value))
|
||
|
if actual != expected {
|
||
|
t.Errorf(
|
||
|
"expected %s, got %s",
|
||
|
expected, actual,
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
expect("foo\\0bar", "foo\x00bar")
|
||
|
expect("foo\\nbar", "foo\nbar")
|
||
|
expect("foo\\rbar", "foo\rbar")
|
||
|
expect("foo\\Zbar", "foo\x1abar")
|
||
|
expect("foo\\\"bar", "foo\"bar")
|
||
|
expect("foo\\\\bar", "foo\\bar")
|
||
|
expect("foo\\'bar", "foo'bar")
|
||
|
}
|
||
|
|
||
|
func TestEscapeQuotes(t *testing.T) {
|
||
|
expect := func(expected, value string) {
|
||
|
actual := string(escapeBytesQuotes([]byte{}, []byte(value)))
|
||
|
if actual != expected {
|
||
|
t.Errorf(
|
||
|
"expected %s, got %s",
|
||
|
expected, actual,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
actual = string(escapeStringQuotes([]byte{}, value))
|
||
|
if actual != expected {
|
||
|
t.Errorf(
|
||
|
"expected %s, got %s",
|
||
|
expected, actual,
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
expect("foo\x00bar", "foo\x00bar") // not affected
|
||
|
expect("foo\nbar", "foo\nbar") // not affected
|
||
|
expect("foo\rbar", "foo\rbar") // not affected
|
||
|
expect("foo\x1abar", "foo\x1abar") // not affected
|
||
|
expect("foo''bar", "foo'bar") // affected
|
||
|
expect("foo\"bar", "foo\"bar") // not affected
|
||
|
}
|
||
|
|
||
|
func TestAtomicBool(t *testing.T) {
|
||
|
var ab atomicBool
|
||
|
if ab.IsSet() {
|
||
|
t.Fatal("Expected value to be false")
|
||
|
}
|
||
|
|
||
|
ab.Set(true)
|
||
|
if ab.value != 1 {
|
||
|
t.Fatal("Set(true) did not set value to 1")
|
||
|
}
|
||
|
if !ab.IsSet() {
|
||
|
t.Fatal("Expected value to be true")
|
||
|
}
|
||
|
|
||
|
ab.Set(true)
|
||
|
if !ab.IsSet() {
|
||
|
t.Fatal("Expected value to be true")
|
||
|
}
|
||
|
|
||
|
ab.Set(false)
|
||
|
if ab.value != 0 {
|
||
|
t.Fatal("Set(false) did not set value to 0")
|
||
|
}
|
||
|
if ab.IsSet() {
|
||
|
t.Fatal("Expected value to be false")
|
||
|
}
|
||
|
|
||
|
ab.Set(false)
|
||
|
if ab.IsSet() {
|
||
|
t.Fatal("Expected value to be false")
|
||
|
}
|
||
|
if ab.TrySet(false) {
|
||
|
t.Fatal("Expected TrySet(false) to fail")
|
||
|
}
|
||
|
if !ab.TrySet(true) {
|
||
|
t.Fatal("Expected TrySet(true) to succeed")
|
||
|
}
|
||
|
if !ab.IsSet() {
|
||
|
t.Fatal("Expected value to be true")
|
||
|
}
|
||
|
|
||
|
ab.Set(true)
|
||
|
if !ab.IsSet() {
|
||
|
t.Fatal("Expected value to be true")
|
||
|
}
|
||
|
if ab.TrySet(true) {
|
||
|
t.Fatal("Expected TrySet(true) to fail")
|
||
|
}
|
||
|
if !ab.TrySet(false) {
|
||
|
t.Fatal("Expected TrySet(false) to succeed")
|
||
|
}
|
||
|
if ab.IsSet() {
|
||
|
t.Fatal("Expected value to be false")
|
||
|
}
|
||
|
|
||
|
ab._noCopy.Lock() // we've "tested" it ¯\_(ツ)_/¯
|
||
|
}
|
||
|
|
||
|
func TestAtomicError(t *testing.T) {
|
||
|
var ae atomicError
|
||
|
if ae.Value() != nil {
|
||
|
t.Fatal("Expected value to be nil")
|
||
|
}
|
||
|
|
||
|
ae.Set(ErrMalformPkt)
|
||
|
if v := ae.Value(); v != ErrMalformPkt {
|
||
|
if v == nil {
|
||
|
t.Fatal("Value is still nil")
|
||
|
}
|
||
|
t.Fatal("Error did not match")
|
||
|
}
|
||
|
ae.Set(ErrPktSync)
|
||
|
if ae.Value() == ErrMalformPkt {
|
||
|
t.Fatal("Error still matches old error")
|
||
|
}
|
||
|
if v := ae.Value(); v != ErrPktSync {
|
||
|
t.Fatal("Error did not match")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestIsolationLevelMapping(t *testing.T) {
|
||
|
data := []struct {
|
||
|
level driver.IsolationLevel
|
||
|
expected string
|
||
|
}{
|
||
|
{
|
||
|
level: driver.IsolationLevel(sql.LevelReadCommitted),
|
||
|
expected: "READ COMMITTED",
|
||
|
},
|
||
|
{
|
||
|
level: driver.IsolationLevel(sql.LevelRepeatableRead),
|
||
|
expected: "REPEATABLE READ",
|
||
|
},
|
||
|
{
|
||
|
level: driver.IsolationLevel(sql.LevelReadUncommitted),
|
||
|
expected: "READ UNCOMMITTED",
|
||
|
},
|
||
|
{
|
||
|
level: driver.IsolationLevel(sql.LevelSerializable),
|
||
|
expected: "SERIALIZABLE",
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for i, td := range data {
|
||
|
if actual, err := mapIsolationLevel(td.level); actual != td.expected || err != nil {
|
||
|
t.Fatal(i, td.expected, actual, err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// check unsupported mapping
|
||
|
expectedErr := "mysql: unsupported isolation level: 7"
|
||
|
actual, err := mapIsolationLevel(driver.IsolationLevel(sql.LevelLinearizable))
|
||
|
if actual != "" || err == nil {
|
||
|
t.Fatal("Expected error on unsupported isolation level")
|
||
|
}
|
||
|
if err.Error() != expectedErr {
|
||
|
t.Fatalf("Expected error to be %q, got %q", expectedErr, err)
|
||
|
}
|
||
|
}
|